Compare commits

..

10 Commits

Author SHA1 Message Date
minori 5ccbde472f deps: update @nx/js to 22.7.3
Node.js CI / CI (pull_request) Failing after 36s
Security Scan and Upload / Security & DefectDojo Upload (pull_request) Successful in 1m39s
2026-06-02 07:05:26 -07:00
minori 11589b1881 deps: update @nx/js to 22.7.2
Node.js CI / CI (pull_request) Failing after 14s
Security Scan and Upload / Security & DefectDojo Upload (pull_request) Successful in 1m51s
2026-05-25 07:05:09 -07:00
minori ef86f281e1 deps: update @nx/js to 22.7.1
Node.js CI / CI (pull_request) Failing after 16s
Security Scan and Upload / Security & DefectDojo Upload (pull_request) Successful in 1m45s
2026-05-09 07:10:09 -07:00
minori 4ec0a220f0 deps: update @nx/js to 22.7.0
Node.js CI / CI (pull_request) Failing after 17s
Security Scan and Upload / Security & DefectDojo Upload (pull_request) Successful in 2m21s
2026-05-05 17:44:39 -07:00
minori 8b5bffb696 deps: update @nx/js to 22.6.5
Node.js CI / CI (pull_request) Failing after 11s
Security Scan and Upload / Security & DefectDojo Upload (pull_request) Successful in 1m38s
2026-04-21 07:04:48 -07:00
minori a4441d9570 deps: update @nx/js to 22.6.4
Node.js CI / CI (pull_request) Failing after 20s
Security Scan and Upload / Security & DefectDojo Upload (pull_request) Successful in 2m1s
2026-04-12 07:04:21 -07:00
minori 2860c8c8d8 deps: update @nx/js to 22.6.3
Node.js CI / CI (pull_request) Failing after 13s
Security Scan and Upload / Security & DefectDojo Upload (pull_request) Successful in 1m37s
2026-04-07 07:04:27 -07:00
minori 90888300e5 deps: update @nx/js to 22.6.2
Node.js CI / CI (pull_request) Failing after 12s
Security Scan and Upload / Security & DefectDojo Upload (pull_request) Successful in 1m21s
2026-04-06 07:04:10 -07:00
minori 483d1708a6 deps: update @nx/js to 22.5.0
Node.js CI / CI (pull_request) Failing after 26s
Security Scan and Upload / Security & DefectDojo Upload (pull_request) Successful in 1m27s
2026-02-20 07:13:53 -08:00
minori baa15b8107 deps: update @nx/js to 22.4.5
Node.js CI / CI (pull_request) Failing after 17s
Security Scan and Upload / Security & DefectDojo Upload (pull_request) Successful in 1m29s
2026-02-14 07:08:59 -08:00
183 changed files with 3782 additions and 26412 deletions
-299
View File
@@ -1,299 +0,0 @@
# Library Project - Claude Instructions
This is a personal media library tracking application built with an Nx monorepo structure. It tracks games, books, music, art, shows, and manga with user profiles, comments, achievements, and social features.
## Git Commit Guidelines
**CRITICAL**: When working on this project:
- **Always commit as Hikari** using: `--author="Hikari <hikari@nhcarrigan.com>" --no-gpg-sign`
- **Always ask permission before creating commits** - Never commit without explicit user approval
- **Never modify git config** - Always use CLI flags for author information
## User Permissions
This is **Naomi's personal library**:
- **Admin (Naomi only)** - Can create, update, and delete all media items (games, books, music, art, shows, manga)
- **Regular users** - Can only:
- Like items
- Comment on items (with markdown support)
- Submit suggestions for new items (requires admin approval)
- Customise their own profile
All CRUD operations on media items are **admin-only**. Regular users interact through the social features (likes, comments, suggestions).
## Project Structure
This is an **Nx monorepo** containing:
- **`api/`** - Fastify backend API with Prisma ORM
- **`apps/frontend/`** - Angular frontend application
- **`shared-types/`** - Shared TypeScript types between frontend and backend
- **`libs/`** - Shared libraries
## Technology Stack
### Backend (api/)
- **Runtime**: Node.js v24
- **Framework**: Fastify 5.x with plugins:
- `@fastify/autoload` - Auto-loads routes and plugins
- `@fastify/jwt` - JWT authentication
- `@fastify/oauth2` - Discord OAuth integration
- `@fastify/helmet` - Security headers
- `@fastify/cors` - CORS configuration
- `@fastify/csrf-protection` - CSRF protection
- `@fastify/rate-limit` - Rate limiting
- `@fastify/cookie` - Cookie handling
- `@fastify/static` - Static file serving
- **Database**: MongoDB with Prisma ORM
- **Logging**: `@nhcarrigan/logger` (custom logger package)
- **Markdown**: `marked` for parsing, `dompurify` + `jsdom` for sanitisation
### Frontend (apps/frontend/)
- **Framework**: Angular 21.x
- **Icons**: FontAwesome (solid + brands)
- **Styling**: Angular's built-in styling system
### Development Tools
- **Package Manager**: pnpm v10
- **Monorepo**: Nx 22.x
- **Linting**: ESLint 9 (flat config) with `@nhcarrigan/eslint-config`
- **Testing**: Jest 30.x
- **Build**: esbuild (backend), Angular CLI (frontend)
- **Secrets**: 1Password CLI (`op run`)
## Database Schema
The Prisma schema (`api/prisma/schema.prisma`) defines these models:
- **Game** - Video game tracking with platform, status, rating, cover image, tags, series
- **Book** - Book tracking with author, ISBN, status, rating, cover image, tags, series
- **Music** - Music tracking (album/single/EP) with artist, status, rating, cover art, tags
- **Art** - Art collection with artist, description, image, tags
- **Show** - TV/anime/film tracking with type, status, rating, cover image, tags
- **Manga** - Manga tracking with author, status, rating, cover image, tags
- **User** - User profiles with Discord auth, slug, bio, badges, achievements, social links
- **Comment** - Comments on any media type with markdown support
- **AuditLog** - Security and action logging
- **Suggestion** - User-submitted content suggestions
- **Like** - User likes on media items
- **RefreshToken** - JWT refresh token storage
- **ProfileReport** - User profile reporting system
- **CommentReport** - Comment reporting system
- **UserAchievement** - Achievement tracking with progress
All models use MongoDB ObjectIds and include timestamps (`createdAt`, `updatedAt`).
## Authentication Flow
The application uses **Discord OAuth** for authentication:
1. User clicks "Login with Discord"
2. OAuth flow redirects to Discord for authorisation
3. Discord redirects back with authorisation code
4. Backend exchanges code for Discord user info
5. Backend creates/updates User record
6. Backend issues JWT access token (15m expiry) and refresh token (7d expiry)
7. Frontend stores tokens in cookies
8. Access token in `Authorization` header for API requests
9. Refresh token used to obtain new access tokens
See `api/AUTH_FLOW.md` for detailed authentication implementation.
## Image Upload Handling
The application supports **two methods** for cover images:
1. **Regular URLs** - Standard image URLs (max 2048 characters)
2. **Base64 Data URLs** - Inline base64-encoded images (max 5MB decoded size)
### Base64 Upload Constraints
- **Fastify body limit**: 10MB (accommodates base64 overhead)
- **Validation**: Data URLs must match format `data:image/(jpeg|png|gif|webp|svg+xml);base64,[base64data]`
- **Size calculation**: Base64 data is extracted and decoded size calculated as `base64Data.length * 0.75`
- **Size limit**: Decoded image must be under 5MB
### Implementation Notes
- Base64 size check happens AFTER format validation
- Regular URL length check (2048 chars) does NOT apply to data URLs
- Validation logic is in `api/src/app/services/*.service.ts` files
- Base64 validation helpers in `api/src/app/utils/validation.ts`
## Development Workflow
### Local Development
```bash
pnpm dev # Builds all projects and starts API with dev.env secrets
```
The `dev` script:
1. Runs `nx run-many --target=build --all` to build all projects
2. Uses `op run --env-file=dev.env` to inject secrets from 1Password
3. Starts the API with `NODE_ENV=production node dist/api/main.js`
4. API serves the frontend as static files from `dist/apps/frontend/browser`
### Separate Frontend/Backend Development
```bash
pnpm start:frontend:dev # nx serve frontend (port 4200)
pnpm start:api:dev # nx serve api (port 3000, watch mode)
```
### Building for Production
```bash
pnpm build # Generates Prisma client, then builds all projects
```
### Database Management
```bash
pnpm db:gen # Generate Prisma client
pnpm db:push # Push schema changes to MongoDB (uses prod.env secrets)
```
### Linting & Testing
```bash
pnpm lint # Lints all projects with ESLint
pnpm test # Runs all tests with Jest
```
## CI/CD Pipeline
The project uses **Gitea Actions** (`.gitea/workflows/ci.yml`):
1. **Dependency Pin Check** - Ensures all dependencies are pinned (no `^` or `~`)
2. **Install Dependencies** - `pnpm install`
3. **Lint** - `pnpm run lint`
4. **Build** - `pnpm run build`
5. **Test** - `pnpm run test`
CI runs on:
- Pushes to `main` branch
- Pull requests to `main` branch
## Configuration Standards
### TypeScript
- Uses `@nhcarrigan/typescript-config` as base
- Separate configs for app code (`tsconfig.app.json`) and tests (`tsconfig.spec.json`)
- Output directory: `dist/` for builds
### ESLint
- Uses `@nhcarrigan/eslint-config` (ESLint 9 flat config)
- **No Prettier** - all style rules handled by ESLint
- Configured in `eslint.config.mjs`
- Angular-specific rules enabled for frontend
### Testing
- Jest 30.x with `ts-jest` for TypeScript support
- Separate jest configs per project (e.g., `api/jest.config.cts`)
- Cypress for e2e testing (frontend-e2e project)
## Secrets Management
### Environment Files
- **`dev.env`** - Development secrets (1Password vault references)
- **`prod.env`** - Production secrets (1Password vault references)
These files contain **ONLY** 1Password references (e.g., `op://vault/item/field`), NOT actual secrets. They are safe to commit.
### Non-Secret Configuration
Configuration that isn't sensitive (URLs, intervals, feature flags) should be in code (e.g., constants files), NOT in `.env` files.
### Running with Secrets
Always use 1Password CLI:
```bash
op run --env-file=dev.env -- <command>
op run --env-file=prod.env -- <command>
```
## Security Features
The application implements multiple security layers:
- **Helmet** - Security headers (CSP, HSTS, etc.)
- **CORS** - Configured origin restrictions
- **CSRF Protection** - Token-based CSRF validation
- **Rate Limiting** - Request rate limits per IP
- **JWT** - Short-lived access tokens with refresh token rotation
- **Audit Logging** - All security events logged to AuditLog model
- **Content Sanitisation** - Markdown comments sanitised with DOMPurify
- **Input Validation** - All inputs validated before database operations
## API Routes Structure
Routes are auto-loaded via `@fastify/autoload` from `api/src/app/routes/`:
- Each route file exports Fastify route handlers
- Routes follow REST conventions (GET, POST, PUT, DELETE)
- All routes require JWT authentication (except auth endpoints)
- Route handlers call service functions for business logic
Example structure:
- `api/src/app/routes/games/index.ts` - Game CRUD endpoints
- `api/src/app/services/game.service.ts` - Game business logic
## Code Style Conventions
### General
- Use **clear, descriptive variable/function names**
- Prefer **self-documenting code** over comments
- Only use comments to explain **reasoning** when intent isn't clear
- Use **British English** spelling (colour, organise, whilst)
### TypeScript
- Prefer `interface` over `type` for object shapes
- Use `const` assertions where appropriate
- Enable strict mode (inherited from `@nhcarrigan/typescript-config`)
### Error Handling
- Service functions throw `Error` instances with descriptive messages
- Route handlers wrap service calls in try-catch blocks
- Return 400 for validation errors with error message
- Return 500 for unexpected errors (message hidden in production)
- All errors logged to AuditLog for security events
### Validation
- Validation utilities in `api/src/app/utils/validation.ts`
- Constants for max lengths in `shared-types/src/lib/constants.ts`
- Validate early in service functions before database operations
## Common Gotchas
### Base64 Image Uploads
- **Body limit must be 10MB** (`api/src/main.ts`) to accommodate base64 overhead
- Validate data URL format BEFORE size check
- Extract base64 portion (after comma) for size calculation
- Don't apply regular URL length validation to data URLs
### Prisma Client Generation
- Must run `pnpm db:gen` before building if schema changed
- The build script automatically runs this
- Prisma client is generated to `api/generated/`
### Nx Cache
- Nx caches build outputs for faster rebuilds
- Clear cache with `nx reset` if experiencing issues
- Cache stored in `.nx/cache/`
### MongoDB ObjectIds
- All IDs are MongoDB ObjectIds (not UUIDs or auto-increment integers)
- Use `@db.ObjectId` in Prisma schema
- Use `String` type in TypeScript for ObjectIds
## Deployment
The application is deployed in production mode:
```bash
pnpm build # Build all projects
pnpm start # Start with prod.env secrets
```
Production start command:
```bash
NODE_ENV=production op run --env-file=prod.env -- node dist/api/main.js
```
The API serves the built frontend as static files, so only the API process needs to run in production.
## Important Notes
- This is a **personal project** for Naomi's media tracking
- Uses **Discord OAuth** for authentication (no other auth methods)
- **MongoDB** is the only supported database (Prisma schema is MongoDB-specific)
- All **timestamps** are stored as UTC in the database
- **Achievements system** tracks user activity and unlocks (see UserAchievement model)
- **Comments support markdown** with sanitisation
- **User profiles** support custom slugs, bios, and social links
- **Reporting system** for profiles and comments with moderation workflow
-458
View File
@@ -1,458 +0,0 @@
# Security Audit Report - Library Application
**Date:** 20 February 2026
**Audited by:** Hikari
**Application:** Library Management System (Books, Games, Music, Art, Shows, Manga)
---
## Executive Summary
A comprehensive security audit was conducted on the library application covering authentication, authorisation, input validation, XSS/CSRF protection, API security, database security, and dependency vulnerabilities. The application demonstrates strong security fundamentals with proper authentication, CSRF protection, and XSS sanitization. However, several critical improvements have been identified and implemented.
**Overall Security Rating:** 8/10 (Improved from 6.5/10)
---
## 1. Authentication & Authorisation ✅
### Strengths
- **Secure JWT Implementation**
- HS256 algorithm used consistently
- Short-lived access tokens (15 minutes)
- JWT secret required via environment variable
- Proper token signing and verification
- **Robust Refresh Token System**
- Cryptographically secure tokens (64 bytes from crypto.randomBytes)
- 7-day expiry with database storage
- Token rotation on each refresh (security best practice)
- Proper cleanup of expired tokens
- Refresh tokens invalidated on ban
- **Proper Authentication Middleware**
- `app.authenticate` decorator consistently applied
- `adminGuard` middleware checks database for fresh admin status (prevents stale JWT claims)
- `bannedGuard` middleware prevents banned users from actions
- **Cookie Security**
- HttpOnly cookies prevent JavaScript access
- Secure flag enabled in production
- SameSite=lax prevents CSRF via cookies
- Signed cookies prevent tampering
- Separate paths for auth-token (/) and refresh-token (/api/auth)
### Areas of Concern (Fixed)
- ✅ Admin status checked from database in middleware (prevents JWT claim staleness)
- ✅ Banned user check prevents actions before token expiry
### Recommendations
- Consider implementing rate limiting on login/refresh endpoints specifically
- Add IP address tracking for suspicious activity detection
- Consider implementing 2FA for admin accounts
**Confidence Score:** 9/10
---
## 2. Input Validation & Sanitization ✅ (IMPROVED)
### Previous Issues (NOW FIXED)
-**No Runtime Validation** - DTOs were TypeScript interfaces only (runtime = any object)
-**URL Validation Missing** - User-provided URLs not validated for dangerous protocols
-**No Length Limits** - Could lead to DoS via extremely long strings
### Improvements Implemented
-**Created Validation Utility** (`/home/naomi/code/naomi/library/api/src/app/utils/validation.ts`)
- `validateUrl()` - Prevents javascript:, data:, vbscript:, file: URLs; only allows http/https
- `validateSlug()` - Alphanumeric, hyphens, underscores only
- `validateRating()` - Integer 0-10 validation
- `validateStringLength()` - Enforces max lengths
- `MAX_LENGTHS` constants for all field types
-**Applied to User Service**
- URL validation for website, discordServer, bluesky, github, linkedin, twitch, youtube
- Slug format validation (prevents XSS via slug)
- Length limits on displayName (100), bio (1000), URLs (2048)
-**Applied to Comment Service**
- Content length validation (10,000 characters max)
- Prevents DoS via massive comments
-**Applied to Book Service**
- Title (500), author (200), notes (5000), ISBN (50) length limits
- Rating validation (0-10)
- Cover image URL validation
- Tag length validation (50 per tag)
- Link URL and title validation
### Remaining Strengths
- **Excellent Markdown Sanitization** (Comment Service)
- DOMPurify with strict allowlist of HTML tags
- Custom hook blocks javascript:, data:, vbscript: in hrefs
- External links get target="_blank" rel="noopener noreferrer nofollow"
- No data attributes allowed
- Forced body mode prevents context-dependent XSS
### Next Steps
- Apply similar validation to Game, Music, Art, Show, Manga services
- Consider adding Zod or similar schema validation library for runtime type checking
- Add validation for date fields (prevent future dates where inappropriate)
**Confidence Score:** 8/10 (Improved from 4/10)
---
## 3. XSS & CSRF Protection ✅
### Strengths
- **CSRF Protection**
- `@fastify/csrf-protection` plugin registered
- CSRF tokens required via X-CSRF-Token header
- Applied to all state-changing routes (POST, PUT, DELETE)
- Cookie-based session plugin
- `/auth/csrf-token` endpoint provides tokens
- **XSS Protection - Backend**
- DOMPurify sanitizes all user comments (see section 2)
- Markdown rendered safely with allowlist
- Dangerous protocols blocked in links
- **XSS Protection - Frontend**
- Angular's DomSanitizer used in `SanitizeService`
- `SecurityContext.HTML` applied before rendering
- Defense-in-depth: both backend and frontend sanitization
- **No innerHTML with Unsanitized Content**
- All `[innerHTML]` usage goes through `sanitizeService.sanitizeHtml()`
- Example: `comment-display.component.ts` line 71
### Areas for Improvement
- CSRF tokens should be rotated more frequently (currently session-based)
- Consider adding CSP nonce for inline scripts if needed in future
**Confidence Score:** 9/10
---
## 4. API Security ✅ (IMPROVED)
### Strengths
- **Rate Limiting**
- `@fastify/rate-limit` plugin active
- 100 requests per minute per IP
- Logged to audit log when exceeded
- Prevents brute force and DoS
- **CORS Configuration**
- Origin restricted to BASE_URL environment variable
- Credentials enabled (required for cookies)
- Methods limited to GET, POST, PUT, DELETE, OPTIONS
- Headers limited to Content-Type, Authorization, X-CSRF-Token
- **Security Headers** (IMPROVED)
- Content Security Policy configured
- **Improved CSP:**
- `styleSrc` only allows unsafe-inline in development (removed in production)
- Added `fontSrc`, `objectSrc`, `baseUri`, `formAction`, `frameAncestors`
- `frameAncestors: 'none'` prevents clickjacking
- **Added HSTS:** 1 year max-age, includeSubDomains, preload
- **X-Frame-Options:** DENY (clickjacking protection)
- **Referrer-Policy:** strict-origin-when-cross-origin
- X-Content-Type-Options: nosniff (via helmet defaults)
- **Error Handling**
- Global error handler prevents stack trace leaks
- 5xx errors return generic message: "An unexpected error occurred"
- 4xx errors return specific messages (safe to expose)
- Security events logged to audit log
- **Audit Logging**
- Comprehensive audit log for security events
- Logs: login, logout, failed login, CSRF failures, rate limit exceeded, unauthorized access
- Includes user agent, IP address, user ID, resource details
### Improvements Implemented
- ✅ Enhanced CSP removes unsafe-inline in production
- ✅ Added HSTS, X-Frame-Options, Referrer-Policy
- ✅ Removed console.error in favour of Fastify logger
**Confidence Score:** 9/10 (Improved from 7/10)
---
## 5. Database Security ✅
### Strengths
- **Prisma ORM Protection**
- All database queries use Prisma
- Prevents SQL/NoSQL injection via parameterized queries
- Type-safe query building
- **MongoDB with Prisma**
- No raw queries found in codebase
- All queries use Prisma's query builder
- ObjectId validation via Prisma schema
- **Access Control**
- Database URL stored in environment variable
- No database credentials in code
- Uses 1Password for secrets management
- **Sensitive Data Protection**
- Passwords not stored (OAuth only)
- Email addresses only visible to authenticated users
- Sensitive fields not logged (using @nhcarrigan/logger)
### No Issues Found
**Confidence Score:** 10/10
---
## 6. Secrets Management ✅
### Strengths
- **1Password CLI Integration**
- All secrets stored in 1Password vault
- `prod.env` and `dev.env` contain only `op://` references
- Safe to commit to version control
- Secrets injected at runtime via `op run`
- **Required Secrets Validated**
- JWT_SECRET required or application fails
- Database URL required via Prisma
- Discord OAuth credentials required for auth plugin
- **No Hardcoded Secrets**
- Comprehensive search found no hardcoded secrets
- All sensitive values use process.env
### No Issues Found
**Confidence Score:** 10/10
---
## 7. Dependency Security ⚠️ (ACTION REQUIRED)
### Critical Vulnerabilities Identified
```
┌─────────────────────┬────────────────────────────────────────────────────────┐
│ high │ @modelcontextprotocol/sdk has cross-client data leak │
│ │ via shared server/transport instance reuse │
├─────────────────────┼────────────────────────────────────────────────────────┤
│ Patched versions │ >=1.26.0 │
└─────────────────────┴────────────────────────────────────────────────────────┘
```
```
┌─────────────────────┬────────────────────────────────────────────────────────┐
│ high │ Arbitrary File Read/Write via Hardlink Target Escape │
│ │ Through Symlink Chain in node-tar Extraction │
├─────────────────────┼────────────────────────────────────────────────────────┤
│ Package │ tar │
├─────────────────────┼────────────────────────────────────────────────────────┤
│ Patched versions │ >=7.5.8 │
└─────────────────────┴────────────────────────────────────────────────────────┘
```
```
┌─────────────────────┬────────────────────────────────────────────────────────┐
│ high │ Axios is Vulnerable to Denial of Service via __proto__ │
│ │ Key in mergeConfig │
├─────────────────────┼────────────────────────────────────────────────────────┤
│ Patched versions │ >=1.13.5 │
└─────────────────────┴────────────────────────────────────────────────────────┘
```
```
┌─────────────────────┬────────────────────────────────────────────────────────┐
│ high │ minimatch has a ReDoS via repeated wildcards with │
│ │ non-matching literal in pattern │
├─────────────────────┼────────────────────────────────────────────────────────┤
│ Patched versions │ >=10.2.1 │
└─────────────────────┴────────────────────────────────────────────────────────┘
```
```
┌─────────────────────┬────────────────────────────────────────────────────────┐
│ high │ Command Injection via Unsanitized `locate` Output in │
│ │ `versions()` — systeminformation │
├─────────────────────┼────────────────────────────────────────────────────────┤
│ Package │ systeminformation │
├─────────────────────┼────────────────────────────────────────────────────────┤
│ Patched versions │ >=5.31.0 │
└─────────────────────┴────────────────────────────────────────────────────────┘
```
```
┌─────────────────────┬────────────────────────────────────────────────────────┐
│ high │ Systeminformation has a Command Injection via │
│ │ unsanitized interface parameter in wifi.js retry path │
├─────────────────────┼────────────────────────────────────────────────────────┤
│ Patched versions │ >=5.30.8 │
└─────────────────────┴────────────────────────────────────────────────────────┘
```
### Recommendations
**CRITICAL:** Update dependencies immediately:
```bash
pnpm update @modelcontextprotocol/sdk tar axios minimatch systeminformation
```
**Impact Assessment:**
- **@modelcontextprotocol/sdk** - Used by Angular CLI (dev dependency only, low runtime risk)
- **tar, axios, minimatch** - Transitive dependencies via Angular CLI/NX (dev dependencies)
- **systeminformation** - Transitive via Cypress (dev dependency, testing only)
While these are development dependencies and don't affect production runtime, they should still be updated to prevent supply chain attacks during development.
**Confidence Score:** 8/10
---
## 8. Additional Security Checks ✅
### Logging Practices (IMPROVED)
- ✅ Uses `@nhcarrigan/logger` service
- ✅ No console.log usage found in production code (FIXED: removed console.error from users route)
- ✅ Fastify's built-in logger used for request logging
- ✅ Error objects logged via structured logging (prevents sensitive data leaks)
### Open Redirect Protection ✅
- OAuth callback redirects to hardcoded "/" path (no user-controlled redirect)
- No redirect parameter acceptance in any route
- BASE_URL environment variable controls OAuth callback URI
### Information Disclosure ✅
- Error messages don't leak internal details (5xx → generic message)
- Stack traces not exposed to clients
- Database errors handled gracefully
- Admin-only routes return 403, not 404 (prevents enumeration but acceptable trade-off)
### HTTPS Enforcement
- Secure cookie flag enabled in production
- HSTS header now configured (NEW)
- Frontend uses relative URLs in production (/api) preventing mixed content
### Session Management ✅
- No long-lived sessions (JWT approach)
- Refresh tokens properly scoped and rotated
- Logout invalidates refresh tokens
- Ban invalidates all user's refresh tokens
**Confidence Score:** 9/10 (Improved from 7/10)
---
## Summary of Improvements Made
### Files Created
1. `/home/naomi/code/naomi/library/api/src/app/utils/validation.ts` - Comprehensive validation utilities
### Files Modified
1. `/home/naomi/code/naomi/library/api/src/app/services/user.service.ts`
- Added URL validation for all social/website links
- Added slug format validation
- Added length limits for displayName, bio, URLs
2. `/home/naomi/code/naomi/library/api/src/app/services/comment.service.ts`
- Added content length validation (10,000 char max)
- Prevents DoS via massive comments
3. `/home/naomi/code/naomi/library/api/src/app/services/book.service.ts`
- Added comprehensive validation for all fields
- URL validation for cover images and links
- Length limits for all string fields
- Rating validation
4. `/home/naomi/code/naomi/library/api/src/app/plugins/helmet.ts`
- Enhanced CSP (removed unsafe-inline in production)
- Added HSTS configuration
- Added X-Frame-Options, Referrer-Policy
- More restrictive security headers
5. `/home/naomi/code/naomi/library/api/src/app/routes/users/index.ts`
- Replaced console.error with Fastify logger
---
## Remaining Action Items
### High Priority
1. **Update Dependencies** (CRITICAL)
```bash
pnpm update @modelcontextprotocol/sdk tar axios minimatch systeminformation
```
2. **Apply Validation to Remaining Services**
- Game Service
- Music Service
- Art Service
- Show Service
- Manga Service
### Medium Priority
3. **Consider Schema Validation Library**
- Evaluate Zod for runtime type checking
- Would catch invalid data before reaching services
- Better developer experience with type inference
4. **Rate Limiting Enhancements**
- Add stricter rate limits on auth endpoints (e.g., 5 login attempts per 15 minutes)
- Add IP-based tracking for suspicious activity
5. **Testing**
- Add integration tests for validation logic
- Test XSS payloads against sanitization
- Test CSRF protection
- Test authentication bypass attempts
### Low Priority
6. **Documentation**
- Document validation rules for frontend developers
- Create security best practices guide
- Document audit log schema
7. **Monitoring**
- Set up alerts for repeated audit log security events
- Monitor rate limit violations
- Track failed login attempts
---
## OWASP Top 10 Coverage
| Vulnerability | Status | Notes |
|--------------|--------|-------|
| A01: Broken Access Control | ✅ PROTECTED | Strong auth, proper middleware, admin checks |
| A02: Cryptographic Failures | ✅ PROTECTED | Secure tokens, HTTPS, signed cookies |
| A03: Injection | ✅ PROTECTED | Prisma ORM, DOMPurify, URL validation |
| A04: Insecure Design | ✅ GOOD | Secure architecture, defense in depth |
| A05: Security Misconfiguration | ⚠️ GOOD | Strong headers, but deps need updates |
| A06: Vulnerable Components | ⚠️ ACTION NEEDED | 6 high-severity vulnerabilities in dev deps |
| A07: Auth Failures | ✅ PROTECTED | Robust JWT + refresh token system |
| A08: Software/Data Integrity | ✅ PROTECTED | 1Password secrets, signed cookies |
| A09: Logging Failures | ✅ GOOD | Comprehensive audit logging |
| A10: SSRF | ✅ PROTECTED | URL validation prevents malicious redirects |
---
## Final Security Score
**Before Audit:** 6.5/10
**After Improvements:** 8.5/10
**After Dependency Updates:** 9/10 (projected)
The application demonstrates strong security fundamentals with excellent authentication, comprehensive CSRF/XSS protection, and proper secrets management. The main improvements needed are:
1. Updating vulnerable dependencies (CRITICAL)
2. Extending validation to remaining services (HIGH)
3. Adding runtime schema validation (MEDIUM)
---
**Confidence Score for Overall Audit:** 9/10
This audit was conducted with thorough analysis of authentication flows, input handling, security headers, database queries, and dependency versions. I am confident in the findings and recommendations.
+1 -5
View File
@@ -3,12 +3,8 @@ module.exports = {
preset: '../jest.preset.js',
testEnvironment: 'node',
transform: {
'^.+\\.[tj]s$': ['ts-jest', {
tsconfig: '<rootDir>/tsconfig.spec.json',
isolatedModules: true,
}],
'^.+\\.[tj]s$': ['ts-jest', { tsconfig: '<rootDir>/tsconfig.spec.json' }],
},
moduleFileExtensions: ['ts', 'js', 'html'],
coverageDirectory: '../coverage/api',
setupFilesAfterEnv: ['<rootDir>/src/test-setup.ts'],
};
+21 -150
View File
@@ -24,17 +24,12 @@ model Game {
platform String?
status GameStatus
dateAdded DateTime @default(now())
dateStarted DateTime?
dateCompleted DateTime?
dateFinished DateTime?
rating Int? @db.Int @default(0)
notes String?
coverImage String?
tags String[]
links Link[]
series String?
seriesOrder Int? @db.Int
timeSpent Int? @db.Int
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
comments Comment[]
@@ -44,7 +39,6 @@ enum GameStatus {
PLAYING
COMPLETED
BACKLOG
RETIRED
}
model Book {
@@ -54,16 +48,12 @@ model Book {
isbn String?
status BookStatus
dateAdded DateTime @default(now())
dateStarted DateTime?
dateFinished DateTime?
rating Int? @db.Int @default(0)
notes String?
coverImage String?
tags String[]
links Link[]
series String?
seriesOrder Int? @db.Int
timeSpent Int? @db.Int
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
comments Comment[]
@@ -73,7 +63,6 @@ enum BookStatus {
READING
FINISHED
TO_READ
RETIRED
}
model Music {
@@ -83,15 +72,12 @@ model Music {
type MusicType
status MusicStatus
dateAdded DateTime @default(now())
dateStarted DateTime?
dateCompleted DateTime?
dateFinished DateTime?
rating Int? @db.Int @default(0)
notes String?
coverArt String?
tags String[]
links Link[]
timeSpent Int? @db.Int
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
comments Comment[]
@@ -107,7 +93,6 @@ enum MusicStatus {
LISTENING
COMPLETED
WANT_TO_LISTEN
RETIRED
}
model Art {
@@ -130,15 +115,12 @@ model Show {
type ShowType
status ShowStatus
dateAdded DateTime @default(now())
dateStarted DateTime?
dateCompleted DateTime?
dateFinished DateTime?
rating Int? @db.Int @default(0)
notes String?
coverImage String?
tags String[]
links Link[]
timeSpent Int? @db.Int
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
comments Comment[]
@@ -155,7 +137,6 @@ enum ShowStatus {
WATCHING
COMPLETED
WANT_TO_WATCH
RETIRED
}
model Manga {
@@ -164,15 +145,12 @@ model Manga {
author String
status MangaStatus
dateAdded DateTime @default(now())
dateStarted DateTime?
dateCompleted DateTime?
dateFinished DateTime?
rating Int? @db.Int @default(0)
notes String?
coverImage String?
tags String[]
links Link[]
timeSpent Int? @db.Int
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
comments Comment[]
@@ -182,14 +160,6 @@ enum MangaStatus {
READING
COMPLETED
WANT_TO_READ
RETIRED
}
enum PrimaryBadge {
STAFF
MOD
VIP
DISCORD
}
model User {
@@ -198,64 +168,40 @@ model User {
username String
email String @unique
avatar String?
slug String?
displayName String?
bio String?
profilePublic Boolean @default(true)
primaryBadge PrimaryBadge?
website String?
discordServer String?
bluesky String?
github String?
linkedin String?
twitch String?
youtube String?
isAdmin Boolean @default(false)
isBanned Boolean @default(false)
inDiscord Boolean @default(false)
isVip Boolean @default(false)
isMod Boolean @default(false)
isStaff Boolean @default(false)
achievementPoints Int @default(0)
currentStreak Int @default(0)
lastStreakCheck DateTime?
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
comments Comment[]
suggestions Suggestion[]
likes Like[]
refreshTokens RefreshToken[]
reportsMade ProfileReport[] @relation("Reporter")
reportsReceived ProfileReport[] @relation("ReportedUser")
reportsReviewed ProfileReport[] @relation("Reviewer")
commentReportsMade CommentReport[] @relation("CommentReporter")
commentReportsReviewed CommentReport[] @relation("CommentReviewer")
userAchievements UserAchievement[]
@@index([slug], map: "User_slug_key")
comments Comment[]
suggestions Suggestion[]
likes Like[]
refreshTokens RefreshToken[]
}
model Comment {
id String @id @default(auto()) @map("_id") @db.ObjectId
id String @id @default(auto()) @map("_id") @db.ObjectId
content String
rawContent String?
userId String @db.ObjectId
user User @relation(fields: [userId], references: [id])
gameId String? @db.ObjectId
game Game? @relation(fields: [gameId], references: [id])
bookId String? @db.ObjectId
book Book? @relation(fields: [bookId], references: [id])
musicId String? @db.ObjectId
music Music? @relation(fields: [musicId], references: [id])
artId String? @db.ObjectId
art Art? @relation(fields: [artId], references: [id])
showId String? @db.ObjectId
show Show? @relation(fields: [showId], references: [id])
mangaId String? @db.ObjectId
manga Manga? @relation(fields: [mangaId], references: [id])
reports CommentReport[]
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
userId String @db.ObjectId
user User @relation(fields: [userId], references: [id])
gameId String? @db.ObjectId
game Game? @relation(fields: [gameId], references: [id])
bookId String? @db.ObjectId
book Book? @relation(fields: [bookId], references: [id])
musicId String? @db.ObjectId
music Music? @relation(fields: [musicId], references: [id])
artId String? @db.ObjectId
art Art? @relation(fields: [artId], references: [id])
showId String? @db.ObjectId
show Show? @relation(fields: [showId], references: [id])
mangaId String? @db.ObjectId
manga Manga? @relation(fields: [mangaId], references: [id])
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
}
model AuditLog {
@@ -289,7 +235,6 @@ enum AuditAction {
RATE_LIMIT_EXCEEDED
CSRF_VALIDATION_FAILED
UNAUTHORIZED_ACCESS
ACHIEVEMENT_UNLOCKED
}
enum AuditCategory {
@@ -357,77 +302,3 @@ model RefreshToken {
@@index([userId])
@@index([expiresAt])
}
enum ReportReason {
INAPPROPRIATE_CONTENT
HARASSMENT
SPAM
IMPERSONATION
OFFENSIVE_NAME
MALICIOUS_LINKS
OTHER
}
enum ReportStatus {
PENDING
REVIEWED
DISMISSED
ACTION_TAKEN
}
model ProfileReport {
id String @id @default(auto()) @map("_id") @db.ObjectId
reportedUserId String @db.ObjectId
reportedUser User @relation("ReportedUser", fields: [reportedUserId], references: [id])
reporterId String @db.ObjectId
reporter User @relation("Reporter", fields: [reporterId], references: [id])
reason ReportReason
details String
status ReportStatus @default(PENDING)
reviewedBy String? @db.ObjectId
reviewer User? @relation("Reviewer", fields: [reviewedBy], references: [id])
reviewNotes String?
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
@@index([reportedUserId])
@@index([reporterId])
@@index([status])
}
model CommentReport {
id String @id @default(auto()) @map("_id") @db.ObjectId
reportedCommentId String @db.ObjectId
reportedComment Comment @relation(fields: [reportedCommentId], references: [id], onDelete: Cascade)
reporterId String @db.ObjectId
reporter User @relation("CommentReporter", fields: [reporterId], references: [id])
reason ReportReason
details String
status ReportStatus @default(PENDING)
reviewedBy String? @db.ObjectId
reviewer User? @relation("CommentReviewer", fields: [reviewedBy], references: [id])
reviewNotes String?
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
@@index([reportedCommentId])
@@index([reporterId])
@@index([status])
}
model UserAchievement {
id String @id @default(auto()) @map("_id") @db.ObjectId
userId String @db.ObjectId
user User @relation(fields: [userId], references: [id])
achievementKey String
progress Int @default(0)
earned Boolean @default(false)
earnedAt DateTime?
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
@@unique([userId, achievementKey])
@@index([userId])
@@index([achievementKey])
@@index([earned])
}
+1 -1
View File
@@ -15,6 +15,6 @@ describe('GET /', () => {
url: '/',
});
expect(response.json()).toEqual({ version: expect.any(String) });
expect(response.json()).toEqual({ message: 'Hello API' });
});
});
+6 -14
View File
@@ -13,8 +13,8 @@ export async function app(fastify: FastifyInstance, opts: AppOptions) {
// Log CSRF validation failures
if (error.code === 'FST_CSRF_INVALID_TOKEN' || error.code === 'FST_CSRF_MISSING_SECRET') {
await AuditService.log({
action: AuditAction.csrfValidationFailed,
category: AuditCategory.security,
action: AuditAction.CSRF_VALIDATION_FAILED,
category: AuditCategory.SECURITY,
details: `CSRF validation failed: ${error.message}, URL: ${request.url}`,
success: false,
}, request).catch(() => {
@@ -22,11 +22,11 @@ export async function app(fastify: FastifyInstance, opts: AppOptions) {
});
}
// Log unauthorized access attempts (exclude /api/auth/me as 401s there are expected during token refresh)
if ((error.statusCode === 401 || error.statusCode === 403) && request.url !== '/api/auth/me') {
// Log unauthorized access attempts
if (error.statusCode === 401 || error.statusCode === 403) {
await AuditService.log({
action: AuditAction.unauthorizedAccess,
category: AuditCategory.security,
action: AuditAction.UNAUTHORIZED_ACCESS,
category: AuditCategory.SECURITY,
details: `Unauthorized access attempt: ${error.message}, URL: ${request.url}`,
success: false,
}, request).catch(() => {
@@ -57,13 +57,5 @@ export async function app(fastify: FastifyInstance, opts: AppOptions) {
fastify.register(AutoLoad, {
dir: path.join(__dirname, 'routes'),
options: { ...opts, prefix: '/api' },
ignorePattern: /root\.ts$/,
});
// Register root route without prefix
fastify.register(AutoLoad, {
dir: path.join(__dirname, 'routes'),
options: { ...opts },
matchFilter: /root\.ts$/,
});
}
+1 -3
View File
@@ -82,9 +82,7 @@ const authPlugin: FastifyPluginAsync = async (app) => {
try {
await request.jwtVerify();
} catch (err) {
const error = new Error("Invalid token");
(error as any).statusCode = 401;
throw error;
throw app.httpErrors.unauthorized("Invalid token");
}
});
};
+1 -19
View File
@@ -13,32 +13,14 @@ const helmetPlugin: FastifyPluginAsync = async (app) => {
contentSecurityPolicy: {
directives: {
defaultSrc: ["'self'"],
// Angular uses inline styles for component encapsulation, so we need to allow them
styleSrc: ["'self'", "'unsafe-inline'", "https://fonts.googleapis.com"],
styleSrc: ["'self'", "'unsafe-inline'"],
imgSrc: ["'self'", "data:", "https:"],
scriptSrc: ["'self'"],
connectSrc: ["'self'", process.env.FRONTEND_URL ?? "http://localhost:4200"],
fontSrc: ["'self'", "data:", "https://fonts.gstatic.com"],
objectSrc: ["'none'"],
baseUri: ["'self'"],
formAction: ["'self'"],
frameAncestors: ["'none'"],
},
},
crossOriginEmbedderPolicy: false,
crossOriginResourcePolicy: { policy: "cross-origin" },
// Add additional security headers
hsts: {
maxAge: 31536000, // 1 year
includeSubDomains: true,
preload: true,
},
frameguard: {
action: "deny",
},
referrerPolicy: {
policy: "strict-origin-when-cross-origin",
},
});
};
+3 -22
View File
@@ -12,32 +12,13 @@ import { AuditAction, AuditCategory } from "@library/shared-types";
const rateLimitPlugin: FastifyPluginAsync = async (app) => {
await app.register(fastifyRateLimit, {
max: async (request) => {
// Try to get user from JWT
try {
await request.jwtVerify();
// Authenticated users get higher limits
return 500;
} catch {
// Unauthenticated users get lower limits
return 100;
}
},
max: 100,
timeWindow: "1 minute",
allowList: async (request) => {
// Bypass rate limiting entirely for admin users
try {
await request.jwtVerify();
return request.user?.isAdmin === true;
} catch {
return false;
}
},
errorResponseBuilder: (request) => {
// Log rate limit exceeded event
AuditService.log({
action: AuditAction.rateLimitExceeded,
category: AuditCategory.security,
action: AuditAction.RATE_LIMIT_EXCEEDED,
category: AuditCategory.SECURITY,
details: `Rate limit exceeded for URL: ${request.url}`,
success: false,
}, request).catch(() => {
+2 -2
View File
@@ -21,8 +21,8 @@ export default async function staticPlugin(app: FastifyInstance) {
// Catch-all route for Angular SPA routing (must be registered after API routes)
app.setNotFoundHandler((request, reply) => {
// Only catch routes that don't start with /api or /assets
if (!request.url.startsWith("/api") && !request.url.startsWith("/assets")) {
// Only catch routes that don't start with /api
if (!request.url.startsWith("/api")) {
reply.sendFile("index.html");
} else {
reply.code(404).send({ error: "Not Found" });
-122
View File
@@ -1,122 +0,0 @@
/**
* @copyright 2026 NHCarrigan
* @license Naomi's Public License
* @author Naomi Carrigan
*/
import type { FastifyPluginAsync } from "fastify";
import {
ACHIEVEMENT_LIST,
ACHIEVEMENTS,
AchievementProgress,
UserAchievementSummary,
} from "@library/shared-types";
import { AchievementService } from "../../services/achievement.service";
const achievementsRoutes: FastifyPluginAsync = async (app) => {
const achievementService = new AchievementService();
/**
* Get all achievement definitions (public route).
*/
app.get("/definitions", async () => {
return ACHIEVEMENT_LIST;
});
/**
* Get a specific achievement definition by key (public route).
*/
app.get<{ Params: { key: string } }>(
"/definitions/:key",
async (request, reply) => {
const { key } = request.params;
const achievement = ACHIEVEMENTS[key];
if (!achievement) {
return reply.notFound("Achievement not found");
}
return achievement;
},
);
/**
* Get current user's achievement summary (authenticated users).
*/
app.get<{ Reply: UserAchievementSummary }>(
"/summary",
{
preValidation: [app.authenticate],
},
async (request) => {
const userId = request.user.id;
const summary = await achievementService.getUserAchievementSummary(
userId,
);
return summary;
},
);
/**
* Get current user's achievement progress (authenticated users).
*/
app.get<{ Reply: AchievementProgress[] }>(
"/progress",
{
preValidation: [app.authenticate],
},
async (request) => {
const userId = request.user.id;
const progress = await achievementService.getUserAchievementProgress(
userId,
);
return progress;
},
);
/**
* Get another user's achievement summary by ID (authenticated users).
*/
app.get<{ Params: { userId: string }; Reply: UserAchievementSummary }>(
"/users/:userId/summary",
{
preValidation: [app.authenticate],
},
async (request, reply) => {
const { userId } = request.params;
try {
const summary = await achievementService.getUserAchievementSummary(
userId,
);
return summary;
} catch (error) {
return reply.notFound("User not found");
}
},
);
/**
* Get another user's achievement progress by ID (authenticated users).
*/
app.get<{ Params: { userId: string }; Reply: AchievementProgress[] }>(
"/users/:userId/progress",
{
preValidation: [app.authenticate],
},
async (request, reply) => {
const { userId } = request.params;
try {
const progress = await achievementService.getUserAchievementProgress(
userId,
);
return progress;
} catch (error) {
return reply.notFound("User not found");
}
},
);
};
export default achievementsRoutes;
-52
View File
@@ -1,52 +0,0 @@
/**
* @copyright 2026 NHCarrigan
* @license Naomi's Public License
* @author Naomi Carrigan
*/
import { FastifyPluginAsync } from "fastify";
import type { ActivityFeedResponse } from "@library/shared-types";
import { ActivityService } from "../../services/activity.service";
const activityRoutes: FastifyPluginAsync = async (app) => {
const activityService = new ActivityService();
/**
* Get activity feed with optional filters.
*/
app.get<{
Querystring: { limit?: number; offset?: number; userId?: string };
Reply: ActivityFeedResponse;
}>("/", async (request) => {
const limit = request.query.limit && request.query.limit > 0
? Math.min(request.query.limit, 100)
: 50;
const offset = request.query.offset && request.query.offset >= 0
? request.query.offset
: 0;
const userId = request.query.userId;
return activityService.getActivityFeed(limit, offset, userId);
});
/**
* Get activity feed for a specific user.
*/
app.get<{
Params: { userId: string };
Querystring: { limit?: number; offset?: number };
Reply: ActivityFeedResponse;
}>("/:userId", async (request) => {
const { userId } = request.params;
const limit = request.query.limit && request.query.limit > 0
? Math.min(request.query.limit, 100)
: 50;
const offset = request.query.offset && request.query.offset >= 0
? request.query.offset
: 0;
return activityService.getActivityFeed(limit, offset, userId);
});
};
export default activityRoutes;
+13 -23
View File
@@ -5,11 +5,10 @@
*/
import { FastifyPluginAsync } from "fastify";
import { Art, CreateArtDto, UpdateArtDto, Comment, CreateCommentDto, AuditAction, AuditCategory, AchievementCategory } from "@library/shared-types";
import { Art, CreateArtDto, UpdateArtDto, Comment, CreateCommentDto, AuditAction, AuditCategory } from "@library/shared-types";
import { ArtService } from "../../services/art.service";
import { CommentService } from "../../services/comment.service";
import { AuditService } from "../../services/audit.service";
import { AchievementService } from "../../services/achievement.service";
import { adminGuard } from "../../middleware/admin-guard";
import { bannedGuard } from "../../middleware/banned-guard";
@@ -47,8 +46,8 @@ const artRoutes: FastifyPluginAsync = async (app) => {
async (request) => {
const art = await artService.createArt(request.body);
await AuditService.logFromRequest(request, {
action: AuditAction.entryCreate,
category: AuditCategory.content,
action: AuditAction.ENTRY_CREATE,
category: AuditCategory.CONTENT,
resourceType: "art",
resourceId: art.id,
details: `Created art: ${art.title}`,
@@ -75,8 +74,8 @@ const artRoutes: FastifyPluginAsync = async (app) => {
const art = await artService.updateArt(id, request.body);
if (art) {
await AuditService.logFromRequest(request, {
action: AuditAction.entryUpdate,
category: AuditCategory.content,
action: AuditAction.ENTRY_UPDATE,
category: AuditCategory.CONTENT,
resourceType: "art",
resourceId: id,
details: `Updated art: ${art.title}`,
@@ -99,8 +98,8 @@ const artRoutes: FastifyPluginAsync = async (app) => {
const { id } = request.params;
await artService.deleteArt(id);
await AuditService.logFromRequest(request, {
action: AuditAction.entryDelete,
category: AuditCategory.content,
action: AuditAction.ENTRY_DELETE,
category: AuditCategory.CONTENT,
resourceType: "art",
resourceId: id,
details: `Deleted art with ID: ${id}`,
@@ -134,21 +133,12 @@ const artRoutes: FastifyPluginAsync = async (app) => {
const userId = request.user.id;
const comment = await commentService.createCommentForArt(id, userId, request.body);
await AuditService.logFromRequest(request, {
action: AuditAction.commentCreate,
category: AuditCategory.content,
action: AuditAction.COMMENT_CREATE,
category: AuditCategory.CONTENT,
resourceType: "art",
resourceId: id,
details: `Added comment to art`,
});
// Check for comment achievements
const achievementService = new AchievementService();
await achievementService.checkAchievements(
userId,
AchievementCategory.Comment,
request
);
return comment;
}
);
@@ -179,8 +169,8 @@ const artRoutes: FastifyPluginAsync = async (app) => {
const comment = await commentService.updateComment(commentId, request.body.content);
await AuditService.logFromRequest(request, {
action: AuditAction.commentUpdate,
category: AuditCategory.content,
action: AuditAction.COMMENT_UPDATE,
category: AuditCategory.CONTENT,
resourceType: "art",
resourceId: id,
details: `Updated comment ${commentId} on art`,
@@ -215,8 +205,8 @@ const artRoutes: FastifyPluginAsync = async (app) => {
await commentService.deleteComment(commentId);
await AuditService.logFromRequest(request, {
action: AuditAction.commentDelete,
category: isAdmin && verification.comment?.userId !== userId ? AuditCategory.admin : AuditCategory.content,
action: AuditAction.COMMENT_DELETE,
category: isAdmin && verification.comment?.userId !== userId ? AuditCategory.ADMIN : AuditCategory.CONTENT,
resourceType: "art",
resourceId: id,
details: `Deleted comment ${commentId} from art`,
+7 -64
View File
@@ -1,8 +1,7 @@
import { FastifyPluginAsync } from "fastify";
import { AuthService } from "../../services/auth.service";
import { AuditService } from "../../services/audit.service";
import { AchievementService } from "../../services/achievement.service";
import { AuthResponse, AuditAction, AuditCategory, AchievementCategory } from "@library/shared-types";
import { AuthResponse, AuditAction, AuditCategory } from "@library/shared-types";
const authRoutes: FastifyPluginAsync = async (app) => {
const authService = new AuthService(app);
@@ -86,69 +85,13 @@ const authRoutes: FastifyPluginAsync = async (app) => {
// Log successful login
await AuditService.log({
action: AuditAction.login,
category: AuditCategory.auth,
action: AuditAction.LOGIN,
category: AuditCategory.AUTH,
userId: user.id,
details: `User ${user.username} logged in via Discord`,
success: true,
}, request);
// Update login streak and check engagement achievements
const achievementService = new AchievementService();
await achievementService.updateLoginStreak(user.id);
await achievementService.checkAchievements(
user.id,
AchievementCategory.Engagement,
request
);
// Assign library member role if user is in Discord server but doesn't have it
const libraryRoleId = process.env.LIBRARY_ROLE_ID;
if (inDiscord && guildId && libraryRoleId) {
try {
const memberResponse = await fetch(
`https://discord.com/api/users/@me/guilds/${guildId}/member`,
{
headers: {
Authorization: `Bearer ${tokenResult.token.access_token}`,
},
}
);
if (memberResponse.ok) {
const memberData = await memberResponse.json() as { roles: string[] };
const hasLibraryRole = memberData.roles.includes(libraryRoleId);
if (!hasLibraryRole) {
const botToken = process.env.DISCORD_BOT_TOKEN;
if (botToken) {
const assignRoleResponse = await fetch(
`https://discord.com/api/v10/guilds/${guildId}/members/${userData.id}/roles/${libraryRoleId}`,
{
method: "PUT",
headers: {
Authorization: `Bot ${botToken}`,
"Content-Type": "application/json",
},
}
);
if (assignRoleResponse.ok || assignRoleResponse.status === 204) {
app.log.info(`Assigned library role to user ${user.username} (${user.id})`);
} else {
app.log.error(
`Failed to assign library role to user ${user.username}: ${assignRoleResponse.status}`
);
}
}
}
}
} catch (error) {
// Don't fail the login if role assignment fails
app.log.error({ err: error }, "Error assigning library role");
}
}
// Set signed cookies and redirect to frontend
reply
.setCookie("auth-token", accessToken, {
@@ -171,8 +114,8 @@ const authRoutes: FastifyPluginAsync = async (app) => {
} catch (error) {
// Log failed login attempt
await AuditService.log({
action: AuditAction.loginFailed,
category: AuditCategory.security,
action: AuditAction.LOGIN_FAILED,
category: AuditCategory.SECURITY,
details: error instanceof Error ? error.message : String(error),
success: false,
}, request);
@@ -286,8 +229,8 @@ const authRoutes: FastifyPluginAsync = async (app) => {
const user = request.user as { id?: string; username?: string };
if (user?.id) {
await AuditService.log({
action: AuditAction.logout,
category: AuditCategory.auth,
action: AuditAction.LOGOUT,
category: AuditCategory.AUTH,
userId: user.id,
details: `User ${user.username ?? "unknown"} logged out`,
success: true,
+13 -34
View File
@@ -5,11 +5,10 @@
*/
import { FastifyPluginAsync } from "fastify";
import { Book, CreateBookDto, UpdateBookDto, Comment, CreateCommentDto, AuditAction, AuditCategory, AchievementCategory } from "@library/shared-types";
import { Book, CreateBookDto, UpdateBookDto, Comment, CreateCommentDto, AuditAction, AuditCategory } from "@library/shared-types";
import { BookService } from "../../services/book.service";
import { CommentService } from "../../services/comment.service";
import { AuditService } from "../../services/audit.service";
import { AchievementService } from "../../services/achievement.service";
import { adminGuard } from "../../middleware/admin-guard";
import { bannedGuard } from "../../middleware/banned-guard";
@@ -24,17 +23,6 @@ const booksRoutes: FastifyPluginAsync = async (app) => {
return bookService.getAllBooks();
});
/**
* Get all books in a series (public route).
*/
app.get<{ Params: { seriesName: string }; Reply: Book[] }>(
"/series/:seriesName",
async (request) => {
const { seriesName } = request.params;
return bookService.getBooksBySeries(seriesName);
}
);
/**
* Get single book by ID (public route).
*/
@@ -58,8 +46,8 @@ const booksRoutes: FastifyPluginAsync = async (app) => {
async (request) => {
const book = await bookService.createBook(request.body);
await AuditService.logFromRequest(request, {
action: AuditAction.entryCreate,
category: AuditCategory.content,
action: AuditAction.ENTRY_CREATE,
category: AuditCategory.CONTENT,
resourceType: "book",
resourceId: book.id,
details: `Created book: ${book.title}`,
@@ -86,8 +74,8 @@ const booksRoutes: FastifyPluginAsync = async (app) => {
const book = await bookService.updateBook(id, request.body);
if (book) {
await AuditService.logFromRequest(request, {
action: AuditAction.entryUpdate,
category: AuditCategory.content,
action: AuditAction.ENTRY_UPDATE,
category: AuditCategory.CONTENT,
resourceType: "book",
resourceId: id,
details: `Updated book: ${book.title}`,
@@ -110,8 +98,8 @@ const booksRoutes: FastifyPluginAsync = async (app) => {
const { id } = request.params;
await bookService.deleteBook(id);
await AuditService.logFromRequest(request, {
action: AuditAction.entryDelete,
category: AuditCategory.content,
action: AuditAction.ENTRY_DELETE,
category: AuditCategory.CONTENT,
resourceType: "book",
resourceId: id,
details: `Deleted book with ID: ${id}`,
@@ -145,21 +133,12 @@ const booksRoutes: FastifyPluginAsync = async (app) => {
const userId = request.user.id;
const comment = await commentService.createCommentForBook(id, userId, request.body);
await AuditService.logFromRequest(request, {
action: AuditAction.commentCreate,
category: AuditCategory.content,
action: AuditAction.COMMENT_CREATE,
category: AuditCategory.CONTENT,
resourceType: "book",
resourceId: id,
details: `Added comment to book`,
});
// Check for comment achievements
const achievementService = new AchievementService();
await achievementService.checkAchievements(
userId,
AchievementCategory.Comment,
request
);
return comment;
}
);
@@ -190,8 +169,8 @@ const booksRoutes: FastifyPluginAsync = async (app) => {
const comment = await commentService.updateComment(commentId, request.body.content);
await AuditService.logFromRequest(request, {
action: AuditAction.commentUpdate,
category: AuditCategory.content,
action: AuditAction.COMMENT_UPDATE,
category: AuditCategory.CONTENT,
resourceType: "book",
resourceId: id,
details: `Updated comment ${commentId} on book`,
@@ -226,8 +205,8 @@ const booksRoutes: FastifyPluginAsync = async (app) => {
await commentService.deleteComment(commentId);
await AuditService.logFromRequest(request, {
action: AuditAction.commentDelete,
category: isAdmin && verification.comment?.userId !== userId ? AuditCategory.admin : AuditCategory.content,
action: AuditAction.COMMENT_DELETE,
category: isAdmin && verification.comment?.userId !== userId ? AuditCategory.ADMIN : AuditCategory.CONTENT,
resourceType: "book",
resourceId: id,
details: `Deleted comment ${commentId} from book`,
-152
View File
@@ -1,152 +0,0 @@
/**
* @copyright NHCarrigan
* @license Naomi's Public License
* @author Naomi Carrigan
*/
import type { FastifyPluginAsync } from "fastify";
import type {
CreateCommentReportDto,
CommentReportWithDetails,
ReportStatus,
UpdateCommentReportDto,
} from "@library/shared-types";
import { ReportReason, AchievementCategory } from "@library/shared-types";
import { CommentReportService } from "../../services/comment-report.service.js";
import { AchievementService } from "../../services/achievement.service";
import { adminGuard } from "../../middleware/admin-guard.js";
const commentReportsRoutes: FastifyPluginAsync = async (fastify) => {
const commentReportService = new CommentReportService();
// Create a new comment report (authenticated users)
fastify.post<{
Body: CreateCommentReportDto;
Reply: CommentReportWithDetails | { error: string };
}>(
"/",
{
preValidation: [fastify.authenticate],
schema: {
body: {
type: "object",
required: ["reportedCommentId", "reason", "details"],
properties: {
reportedCommentId: { type: "string" },
reason: {
type: "string",
enum: Object.values(ReportReason),
},
details: { type: "string", minLength: 10, maxLength: 1000 },
},
},
},
},
async (request, reply) => {
try {
const report = await commentReportService.createReport(
request.user.id,
request.body,
);
return reply.status(201).send(report);
} catch (error) {
if (
error instanceof Error &&
(error.message.includes("already have a pending report") ||
error.message.includes("maximum number of pending reports"))
) {
return reply.status(409).send({ error: error.message });
}
throw error;
}
},
);
// Get all comment reports (admin only)
fastify.get<{
Querystring: { status?: ReportStatus };
Reply: CommentReportWithDetails[];
}>(
"/",
{
preValidation: [fastify.authenticate, adminGuard],
schema: {
querystring: {
type: "object",
properties: {
status: { type: "string" },
},
},
},
},
async (request, reply) => {
const reports = await commentReportService.getAllReports(
request.query.status,
);
return reply.send(reports);
},
);
// Get a single comment report by ID (admin only)
fastify.get<{
Params: { id: string };
Reply: CommentReportWithDetails | { error: string };
}>(
"/:id",
{
preValidation: [fastify.authenticate, adminGuard],
},
async (request, reply) => {
const report = await commentReportService.getReportById(request.params.id);
if (!report) {
return reply.status(404).send({ error: "Report not found" });
}
return reply.send(report);
},
);
// Update a comment report (admin only)
fastify.put<{
Params: { id: string };
Body: UpdateCommentReportDto;
Reply: CommentReportWithDetails;
}>(
"/:id",
{
preValidation: [fastify.authenticate, adminGuard],
schema: {
body: {
type: "object",
required: ["status"],
properties: {
status: { type: "string" },
reviewNotes: { type: "string", maxLength: 1000 },
},
},
},
},
async (request, reply) => {
const report = await commentReportService.updateReport(
request.params.id,
request.user.id,
request.body,
);
// Check for report achievements for the original reporter
if (report.status === "ACTION_TAKEN" || report.status === "DISMISSED") {
const achievementService = new AchievementService();
await achievementService.checkAchievements(
report.reporterId,
AchievementCategory.Report,
request
);
}
return reply.send(report);
},
);
};
export default commentReportsRoutes;
-72
View File
@@ -1,72 +0,0 @@
/**
* @copyright 2026 NHCarrigan
* @license Naomi's Public License
* @author Naomi Carrigan
*/
import { FastifyPluginAsync } from "fastify";
import { Comment, AuditAction, AuditCategory } from "@library/shared-types";
import { CommentService } from "../../services/comment.service";
import { AuditService } from "../../services/audit.service";
import { adminGuard } from "../../middleware/admin-guard";
interface UpdateCommentBody {
content: string;
}
const commentsRoutes: FastifyPluginAsync = async (app) => {
const commentService = new CommentService();
// Admin: Update any comment by ID
app.put<{ Params: { id: string }; Body: UpdateCommentBody; Reply: Comment | { error: string } }>(
"/:id",
{
preValidation: [app.authenticate, adminGuard],
preHandler: [app.csrfProtection],
},
async (request, reply) => {
const { id } = request.params;
const { content } = request.body;
const existingComment = await commentService.getCommentById(id);
if (!existingComment) {
return reply.code(404).send({ error: "Comment not found" });
}
const comment = await commentService.updateComment(id, content);
await AuditService.logFromRequest(request, {
action: AuditAction.commentUpdate,
category: AuditCategory.admin,
details: `Admin updated comment ${id}`,
});
return comment;
}
);
// Admin: Delete any comment by ID
app.delete<{ Params: { id: string }; Reply: { success: boolean } | { error: string } }>(
"/:id",
{
preValidation: [app.authenticate, adminGuard],
preHandler: [app.csrfProtection],
},
async (request, reply) => {
const { id } = request.params;
const existingComment = await commentService.getCommentById(id);
if (!existingComment) {
return reply.code(404).send({ error: "Comment not found" });
}
await commentService.deleteComment(id);
await AuditService.logFromRequest(request, {
action: AuditAction.commentDelete,
category: AuditCategory.admin,
details: `Admin deleted comment ${id}`,
});
return { success: true };
}
);
};
export default commentsRoutes;
+33 -66
View File
@@ -5,11 +5,10 @@
*/
import { FastifyPluginAsync } from "fastify";
import { Game, CreateGameDto, UpdateGameDto, Comment, CreateCommentDto, AuditAction, AuditCategory, AchievementCategory } from "@library/shared-types";
import { Game, CreateGameDto, UpdateGameDto, Comment, CreateCommentDto, AuditAction, AuditCategory } from "@library/shared-types";
import { GameService } from "../../services/game.service";
import { CommentService } from "../../services/comment.service";
import { AuditService } from "../../services/audit.service";
import { AchievementService } from "../../services/achievement.service";
import { adminGuard } from "../../middleware/admin-guard";
import { bannedGuard } from "../../middleware/banned-guard";
@@ -22,15 +21,6 @@ const gamesRoutes: FastifyPluginAsync = async (app) => {
return gameService.getAllGames();
});
// Get all games in a series (public route)
app.get<{ Params: { seriesName: string }; Reply: Game[] }>(
"/series/:seriesName",
async (request) => {
const { seriesName } = request.params;
return gameService.getGamesBySeries(seriesName);
}
);
// Get single game (public route)
app.get<{ Params: { id: string }; Reply: Game | null }>(
"/:id",
@@ -41,29 +31,22 @@ const gamesRoutes: FastifyPluginAsync = async (app) => {
);
// Create game (protected admin route)
app.post<{ Body: CreateGameDto; Reply: Game | { error: string } }>(
app.post<{ Body: CreateGameDto; Reply: Game }>(
"/",
{
preValidation: [app.authenticate, adminGuard],
preHandler: [app.csrfProtection],
},
async (request, reply) => {
try {
const game = await gameService.createGame(request.body);
await AuditService.logFromRequest(request, {
action: AuditAction.entryCreate,
category: AuditCategory.content,
resourceType: "game",
resourceId: game.id,
details: `Created game: ${game.title}`,
});
return game;
} catch (error) {
if (error instanceof Error) {
return reply.code(400).send({ error: error.message });
}
throw error;
}
async (request) => {
const game = await gameService.createGame(request.body);
await AuditService.logFromRequest(request, {
action: AuditAction.ENTRY_CREATE,
category: AuditCategory.CONTENT,
resourceType: "game",
resourceId: game.id,
details: `Created game: ${game.title}`,
});
return game;
}
);
@@ -71,33 +54,26 @@ const gamesRoutes: FastifyPluginAsync = async (app) => {
app.put<{
Params: { id: string };
Body: UpdateGameDto;
Reply: Game | null | { error: string };
Reply: Game | null;
}>(
"/:id",
{
preValidation: [app.authenticate, adminGuard],
preHandler: [app.csrfProtection],
},
async (request, reply) => {
try {
const { id } = request.params;
const game = await gameService.updateGame(id, request.body);
if (game) {
await AuditService.logFromRequest(request, {
action: AuditAction.entryUpdate,
category: AuditCategory.content,
resourceType: "game",
resourceId: id,
details: `Updated game: ${game.title}`,
});
}
return game;
} catch (error) {
if (error instanceof Error) {
return reply.code(400).send({ error: error.message });
}
throw error;
async (request) => {
const { id } = request.params;
const game = await gameService.updateGame(id, request.body);
if (game) {
await AuditService.logFromRequest(request, {
action: AuditAction.ENTRY_UPDATE,
category: AuditCategory.CONTENT,
resourceType: "game",
resourceId: id,
details: `Updated game: ${game.title}`,
});
}
return game;
}
);
@@ -112,8 +88,8 @@ const gamesRoutes: FastifyPluginAsync = async (app) => {
const { id } = request.params;
await gameService.deleteGame(id);
await AuditService.logFromRequest(request, {
action: AuditAction.entryDelete,
category: AuditCategory.content,
action: AuditAction.ENTRY_DELETE,
category: AuditCategory.CONTENT,
resourceType: "game",
resourceId: id,
details: `Deleted game with ID: ${id}`,
@@ -143,21 +119,12 @@ const gamesRoutes: FastifyPluginAsync = async (app) => {
const userId = request.user.id;
const comment = await commentService.createCommentForGame(id, userId, request.body);
await AuditService.logFromRequest(request, {
action: AuditAction.commentCreate,
category: AuditCategory.content,
action: AuditAction.COMMENT_CREATE,
category: AuditCategory.CONTENT,
resourceType: "game",
resourceId: id,
details: `Added comment to game`,
});
// Check for comment achievements
const achievementService = new AchievementService();
await achievementService.checkAchievements(
userId,
AchievementCategory.Comment,
request
);
return comment;
}
);
@@ -186,8 +153,8 @@ const gamesRoutes: FastifyPluginAsync = async (app) => {
const comment = await commentService.updateComment(commentId, request.body.content);
await AuditService.logFromRequest(request, {
action: AuditAction.commentUpdate,
category: AuditCategory.content,
action: AuditAction.COMMENT_UPDATE,
category: AuditCategory.CONTENT,
resourceType: "game",
resourceId: id,
details: `Updated comment ${commentId} on game`,
@@ -220,8 +187,8 @@ const gamesRoutes: FastifyPluginAsync = async (app) => {
await commentService.deleteComment(commentId);
await AuditService.logFromRequest(request, {
action: AuditAction.commentDelete,
category: isAdmin && verification.comment?.userId !== userId ? AuditCategory.admin : AuditCategory.content,
action: AuditAction.COMMENT_DELETE,
category: isAdmin && verification.comment?.userId !== userId ? AuditCategory.ADMIN : AuditCategory.CONTENT,
resourceType: "game",
resourceId: id,
details: `Deleted comment ${commentId} from game`,
-85
View File
@@ -1,85 +0,0 @@
/**
* @copyright 2026 NHCarrigan
* @license Naomi's Public License
* @author Naomi Carrigan
*/
import { FastifyPluginAsync } from "fastify";
import type {
LeaderboardResponse,
SuggestionsLeaderboard,
LikesLeaderboard,
CommentsLeaderboard,
OverallLeaderboard,
} from "@library/shared-types";
import { LeaderboardService } from "../../services/leaderboard.service";
const leaderboardRoutes: FastifyPluginAsync = async (app) => {
const leaderboardService = new LeaderboardService();
/**
* Get all leaderboards at once.
*/
app.get<{
Querystring: { limit?: number };
Reply: LeaderboardResponse;
}>("/", async (request) => {
const limit = request.query.limit && request.query.limit > 0
? Math.min(request.query.limit, 100)
: 25;
return leaderboardService.getAllLeaderboards(limit);
});
/**
* Get top users by suggestions.
*/
app.get<{
Querystring: { limit?: number };
Reply: SuggestionsLeaderboard[];
}>("/suggestions", async (request) => {
const limit = request.query.limit && request.query.limit > 0
? Math.min(request.query.limit, 100)
: 25;
return leaderboardService.getTopSuggestions(limit);
});
/**
* Get top users by likes.
*/
app.get<{
Querystring: { limit?: number };
Reply: LikesLeaderboard[];
}>("/likes", async (request) => {
const limit = request.query.limit && request.query.limit > 0
? Math.min(request.query.limit, 100)
: 25;
return leaderboardService.getTopLikes(limit);
});
/**
* Get top users by comments.
*/
app.get<{
Querystring: { limit?: number };
Reply: CommentsLeaderboard[];
}>("/comments", async (request) => {
const limit = request.query.limit && request.query.limit > 0
? Math.min(request.query.limit, 100)
: 25;
return leaderboardService.getTopComments(limit);
});
/**
* Get overall leaderboard.
*/
app.get<{
Querystring: { limit?: number };
Reply: OverallLeaderboard[];
}>("/overall", async (request) => {
const limit = request.query.limit && request.query.limit > 0
? Math.min(request.query.limit, 100)
: 25;
return leaderboardService.getOverallLeaderboard(limit);
});
};
export default leaderboardRoutes;
-41
View File
@@ -1,41 +0,0 @@
/**
* @copyright 2026 NHCarrigan
* @license Naomi's Public License
* @author Naomi Carrigan
*/
import { FastifyInstance, FastifyRequest } from 'fastify';
import { logger } from '../../utils/logger';
interface LogBody {
level: 'debug' | 'info' | 'warn' | 'error';
message: string;
context?: string;
error?: {
name: string;
message: string;
stack?: string;
};
}
export default async function (fastify: FastifyInstance) {
fastify.post('/', async function (request: FastifyRequest<{ Body: LogBody }>) {
const { level, message, context, error } = request.body;
if (level === 'error' && error) {
const errorObj = new Error(error.message);
errorObj.name = error.name;
if (error.stack) {
errorObj.stack = error.stack;
}
await logger.error(context || 'Frontend', errorObj);
} else if (level === 'error') {
await logger.error('Frontend', new Error(message));
} else {
const logMessage = context ? `[${context}] ${message}` : message;
await logger.log(level, logMessage);
}
return { success: true };
});
}
+13 -23
View File
@@ -5,11 +5,10 @@
*/
import { FastifyPluginAsync } from "fastify";
import { Manga, CreateMangaDto, UpdateMangaDto, Comment, CreateCommentDto, AuditAction, AuditCategory, AchievementCategory } from "@library/shared-types";
import { Manga, CreateMangaDto, UpdateMangaDto, Comment, CreateCommentDto, AuditAction, AuditCategory } from "@library/shared-types";
import { MangaService } from "../../services/manga.service";
import { CommentService } from "../../services/comment.service";
import { AuditService } from "../../services/audit.service";
import { AchievementService } from "../../services/achievement.service";
import { adminGuard } from "../../middleware/admin-guard";
import { bannedGuard } from "../../middleware/banned-guard";
@@ -38,8 +37,8 @@ const mangaRoutes: FastifyPluginAsync = async (app) => {
async (request) => {
const manga = await mangaService.createManga(request.body);
await AuditService.logFromRequest(request, {
action: AuditAction.entryCreate,
category: AuditCategory.content,
action: AuditAction.ENTRY_CREATE,
category: AuditCategory.CONTENT,
resourceType: "manga",
resourceId: manga.id,
details: `Created manga: ${manga.title}`,
@@ -63,8 +62,8 @@ const mangaRoutes: FastifyPluginAsync = async (app) => {
const manga = await mangaService.updateManga(id, request.body);
if (manga) {
await AuditService.logFromRequest(request, {
action: AuditAction.entryUpdate,
category: AuditCategory.content,
action: AuditAction.ENTRY_UPDATE,
category: AuditCategory.CONTENT,
resourceType: "manga",
resourceId: id,
details: `Updated manga: ${manga.title}`,
@@ -84,8 +83,8 @@ const mangaRoutes: FastifyPluginAsync = async (app) => {
const { id } = request.params;
await mangaService.deleteManga(id);
await AuditService.logFromRequest(request, {
action: AuditAction.entryDelete,
category: AuditCategory.content,
action: AuditAction.ENTRY_DELETE,
category: AuditCategory.CONTENT,
resourceType: "manga",
resourceId: id,
details: `Deleted manga with ID: ${id}`,
@@ -113,21 +112,12 @@ const mangaRoutes: FastifyPluginAsync = async (app) => {
const userId = request.user.id;
const comment = await commentService.createCommentForManga(id, userId, request.body);
await AuditService.logFromRequest(request, {
action: AuditAction.commentCreate,
category: AuditCategory.content,
action: AuditAction.COMMENT_CREATE,
category: AuditCategory.CONTENT,
resourceType: "manga",
resourceId: id,
details: `Added comment to manga`,
});
// Check for comment achievements
const achievementService = new AchievementService();
await achievementService.checkAchievements(
userId,
AchievementCategory.Comment,
request
);
return comment;
}
);
@@ -155,8 +145,8 @@ const mangaRoutes: FastifyPluginAsync = async (app) => {
const comment = await commentService.updateComment(commentId, request.body.content);
await AuditService.logFromRequest(request, {
action: AuditAction.commentUpdate,
category: AuditCategory.content,
action: AuditAction.COMMENT_UPDATE,
category: AuditCategory.CONTENT,
resourceType: "manga",
resourceId: id,
details: `Updated comment ${commentId} on manga`,
@@ -188,8 +178,8 @@ const mangaRoutes: FastifyPluginAsync = async (app) => {
await commentService.deleteComment(commentId);
await AuditService.logFromRequest(request, {
action: AuditAction.commentDelete,
category: isAdmin && verification.comment?.userId !== userId ? AuditCategory.admin : AuditCategory.content,
action: AuditAction.COMMENT_DELETE,
category: isAdmin && verification.comment?.userId !== userId ? AuditCategory.ADMIN : AuditCategory.CONTENT,
resourceType: "manga",
resourceId: id,
details: `Deleted comment ${commentId} from manga`,
+13 -23
View File
@@ -5,11 +5,10 @@
*/
import { FastifyPluginAsync } from "fastify";
import { Music, CreateMusicDto, UpdateMusicDto, Comment, CreateCommentDto, AuditAction, AuditCategory, AchievementCategory } from "@library/shared-types";
import { Music, CreateMusicDto, UpdateMusicDto, Comment, CreateCommentDto, AuditAction, AuditCategory } from "@library/shared-types";
import { MusicService } from "../../services/music.service";
import { CommentService } from "../../services/comment.service";
import { AuditService } from "../../services/audit.service";
import { AchievementService } from "../../services/achievement.service";
import { adminGuard } from "../../middleware/admin-guard";
import { bannedGuard } from "../../middleware/banned-guard";
@@ -47,8 +46,8 @@ const musicRoutes: FastifyPluginAsync = async (app) => {
async (request) => {
const music = await musicService.createMusic(request.body);
await AuditService.logFromRequest(request, {
action: AuditAction.entryCreate,
category: AuditCategory.content,
action: AuditAction.ENTRY_CREATE,
category: AuditCategory.CONTENT,
resourceType: "music",
resourceId: music.id,
details: `Created music: ${music.title}`,
@@ -75,8 +74,8 @@ const musicRoutes: FastifyPluginAsync = async (app) => {
const music = await musicService.updateMusic(id, request.body);
if (music) {
await AuditService.logFromRequest(request, {
action: AuditAction.entryUpdate,
category: AuditCategory.content,
action: AuditAction.ENTRY_UPDATE,
category: AuditCategory.CONTENT,
resourceType: "music",
resourceId: id,
details: `Updated music: ${music.title}`,
@@ -99,8 +98,8 @@ const musicRoutes: FastifyPluginAsync = async (app) => {
const { id } = request.params;
await musicService.deleteMusic(id);
await AuditService.logFromRequest(request, {
action: AuditAction.entryDelete,
category: AuditCategory.content,
action: AuditAction.ENTRY_DELETE,
category: AuditCategory.CONTENT,
resourceType: "music",
resourceId: id,
details: `Deleted music with ID: ${id}`,
@@ -134,21 +133,12 @@ const musicRoutes: FastifyPluginAsync = async (app) => {
const userId = request.user.id;
const comment = await commentService.createCommentForMusic(id, userId, request.body);
await AuditService.logFromRequest(request, {
action: AuditAction.commentCreate,
category: AuditCategory.content,
action: AuditAction.COMMENT_CREATE,
category: AuditCategory.CONTENT,
resourceType: "music",
resourceId: id,
details: `Added comment to music`,
});
// Check for comment achievements
const achievementService = new AchievementService();
await achievementService.checkAchievements(
userId,
AchievementCategory.Comment,
request
);
return comment;
}
);
@@ -179,8 +169,8 @@ const musicRoutes: FastifyPluginAsync = async (app) => {
const comment = await commentService.updateComment(commentId, request.body.content);
await AuditService.logFromRequest(request, {
action: AuditAction.commentUpdate,
category: AuditCategory.content,
action: AuditAction.COMMENT_UPDATE,
category: AuditCategory.CONTENT,
resourceType: "music",
resourceId: id,
details: `Updated comment ${commentId} on music`,
@@ -215,8 +205,8 @@ const musicRoutes: FastifyPluginAsync = async (app) => {
await commentService.deleteComment(commentId);
await AuditService.logFromRequest(request, {
action: AuditAction.commentDelete,
category: isAdmin && verification.comment?.userId !== userId ? AuditCategory.admin : AuditCategory.content,
action: AuditAction.COMMENT_DELETE,
category: isAdmin && verification.comment?.userId !== userId ? AuditCategory.ADMIN : AuditCategory.CONTENT,
resourceType: "music",
resourceId: id,
details: `Deleted comment ${commentId} from music`,
-151
View File
@@ -1,151 +0,0 @@
/**
* @copyright NHCarrigan
* @license Naomi's Public License
* @author Naomi Carrigan
*/
import type { FastifyPluginAsync } from "fastify";
import type {
CreateReportDto,
ProfileReportWithUsers,
ReportStatus,
UpdateReportDto,
} from "@library/shared-types";
import { ReportReason, AchievementCategory } from "@library/shared-types";
import { ReportService } from "../../services/report.service.js";
import { AchievementService } from "../../services/achievement.service";
import { adminGuard } from "../../middleware/admin-guard.js";
const reportsRoutes: FastifyPluginAsync = async (fastify) => {
const reportService = new ReportService();
// Create a new report (authenticated users)
fastify.post<{
Body: CreateReportDto;
Reply: ProfileReportWithUsers | { error: string };
}>(
"/",
{
preValidation: [fastify.authenticate],
schema: {
body: {
type: "object",
required: ["reportedUserId", "reason", "details"],
properties: {
reportedUserId: { type: "string" },
reason: {
type: "string",
enum: Object.values(ReportReason),
},
details: { type: "string", minLength: 10, maxLength: 1000 },
},
},
},
},
async (request, reply) => {
try {
const report = await reportService.createReport(
request.user.id,
request.body,
);
return reply.status(201).send(report);
} catch (error) {
if (
error instanceof Error &&
error.message.includes("already have a pending report")
) {
return reply.status(409).send({ error: error.message });
}
throw error;
}
},
);
// Get all reports (admin only)
fastify.get<{
Querystring: { status?: ReportStatus };
Reply: ProfileReportWithUsers[];
}>(
"/",
{
preValidation: [fastify.authenticate, adminGuard],
schema: {
querystring: {
type: "object",
properties: {
status: { type: "string" },
},
},
},
},
async (request, reply) => {
const reports = await reportService.getAllReports(
request.query.status,
);
return reply.send(reports);
},
);
// Get a single report by ID (admin only)
fastify.get<{
Params: { id: string };
Reply: ProfileReportWithUsers | { error: string };
}>(
"/:id",
{
preValidation: [fastify.authenticate, adminGuard],
},
async (request, reply) => {
const report = await reportService.getReportById(request.params.id);
if (!report) {
return reply.status(404).send({ error: "Report not found" });
}
return reply.send(report);
},
);
// Update a report (admin only)
fastify.put<{
Params: { id: string };
Body: UpdateReportDto;
Reply: ProfileReportWithUsers;
}>(
"/:id",
{
preValidation: [fastify.authenticate, adminGuard],
schema: {
body: {
type: "object",
required: ["status"],
properties: {
status: { type: "string" },
reviewNotes: { type: "string", maxLength: 1000 },
},
},
},
},
async (request, reply) => {
const report = await reportService.updateReport(
request.params.id,
request.user.id,
request.body,
);
// Check for report achievements for the original reporter
if (report.status === "ACTION_TAKEN" || report.status === "DISMISSED") {
const achievementService = new AchievementService();
await achievementService.checkAchievements(
report.reporterId,
AchievementCategory.Report,
request
);
}
return reply.send(report);
},
);
};
export default reportsRoutes;
+1 -24
View File
@@ -1,30 +1,7 @@
import { FastifyInstance } from 'fastify';
import { readFileSync } from 'fs';
import { join } from 'path';
interface PackageJson {
version: string;
}
let cachedVersion: string | null = null;
function getVersion(): string {
if (cachedVersion) {
return cachedVersion;
}
try {
const packageJsonPath = join(process.cwd(), 'package.json');
const packageJson = JSON.parse(readFileSync(packageJsonPath, 'utf-8')) as PackageJson;
cachedVersion = packageJson.version;
return cachedVersion;
} catch {
return 'unknown';
}
}
export default async function (fastify: FastifyInstance) {
fastify.get('/', async function () {
return { version: getVersion() };
return { message: 'Hello API' };
});
}
+13 -23
View File
@@ -5,11 +5,10 @@
*/
import { FastifyPluginAsync } from "fastify";
import { Show, CreateShowDto, UpdateShowDto, Comment, CreateCommentDto, AuditAction, AuditCategory, AchievementCategory } from "@library/shared-types";
import { Show, CreateShowDto, UpdateShowDto, Comment, CreateCommentDto, AuditAction, AuditCategory } from "@library/shared-types";
import { ShowService } from "../../services/show.service";
import { CommentService } from "../../services/comment.service";
import { AuditService } from "../../services/audit.service";
import { AchievementService } from "../../services/achievement.service";
import { adminGuard } from "../../middleware/admin-guard";
import { bannedGuard } from "../../middleware/banned-guard";
@@ -38,8 +37,8 @@ const showsRoutes: FastifyPluginAsync = async (app) => {
async (request) => {
const show = await showService.createShow(request.body);
await AuditService.logFromRequest(request, {
action: AuditAction.entryCreate,
category: AuditCategory.content,
action: AuditAction.ENTRY_CREATE,
category: AuditCategory.CONTENT,
resourceType: "show",
resourceId: show.id,
details: `Created show: ${show.title}`,
@@ -63,8 +62,8 @@ const showsRoutes: FastifyPluginAsync = async (app) => {
const show = await showService.updateShow(id, request.body);
if (show) {
await AuditService.logFromRequest(request, {
action: AuditAction.entryUpdate,
category: AuditCategory.content,
action: AuditAction.ENTRY_UPDATE,
category: AuditCategory.CONTENT,
resourceType: "show",
resourceId: id,
details: `Updated show: ${show.title}`,
@@ -84,8 +83,8 @@ const showsRoutes: FastifyPluginAsync = async (app) => {
const { id } = request.params;
await showService.deleteShow(id);
await AuditService.logFromRequest(request, {
action: AuditAction.entryDelete,
category: AuditCategory.content,
action: AuditAction.ENTRY_DELETE,
category: AuditCategory.CONTENT,
resourceType: "show",
resourceId: id,
details: `Deleted show with ID: ${id}`,
@@ -113,21 +112,12 @@ const showsRoutes: FastifyPluginAsync = async (app) => {
const userId = request.user.id;
const comment = await commentService.createCommentForShow(id, userId, request.body);
await AuditService.logFromRequest(request, {
action: AuditAction.commentCreate,
category: AuditCategory.content,
action: AuditAction.COMMENT_CREATE,
category: AuditCategory.CONTENT,
resourceType: "show",
resourceId: id,
details: `Added comment to show`,
});
// Check for comment achievements
const achievementService = new AchievementService();
await achievementService.checkAchievements(
userId,
AchievementCategory.Comment,
request
);
return comment;
}
);
@@ -155,8 +145,8 @@ const showsRoutes: FastifyPluginAsync = async (app) => {
const comment = await commentService.updateComment(commentId, request.body.content);
await AuditService.logFromRequest(request, {
action: AuditAction.commentUpdate,
category: AuditCategory.content,
action: AuditAction.COMMENT_UPDATE,
category: AuditCategory.CONTENT,
resourceType: "show",
resourceId: id,
details: `Updated comment ${commentId} on show`,
@@ -188,8 +178,8 @@ const showsRoutes: FastifyPluginAsync = async (app) => {
await commentService.deleteComment(commentId);
await AuditService.logFromRequest(request, {
action: AuditAction.commentDelete,
category: isAdmin && verification.comment?.userId !== userId ? AuditCategory.admin : AuditCategory.content,
action: AuditAction.COMMENT_DELETE,
category: isAdmin && verification.comment?.userId !== userId ? AuditCategory.ADMIN : AuditCategory.CONTENT,
resourceType: "show",
resourceId: id,
details: `Deleted comment ${commentId} from show`,
+11 -36
View File
@@ -1,8 +1,7 @@
import type { FastifyInstance } from "fastify";
import { SuggestionService } from "../../services/suggestion.service";
import { AuditService } from "../../services/audit.service";
import { AchievementService } from "../../services/achievement.service";
import { AuditAction, AuditCategory, AchievementCategory } from "@library/shared-types";
import { AuditAction, AuditCategory } from "@library/shared-types";
import type {
SuggestionStatus,
SuggestionEntity,
@@ -86,22 +85,14 @@ export default async function (app: FastifyInstance): Promise<void> {
);
await AuditService.logFromRequest(request, {
action: AuditAction.entryCreate,
category: AuditCategory.content,
action: AuditAction.ENTRY_CREATE,
category: AuditCategory.CONTENT,
resourceType: "Suggestion",
resourceId: suggestion.id,
details: `Created ${suggestion.entityType} suggestion: ${suggestion.title}`,
success: true,
});
// Check for suggestion achievements
const achievementService = new AchievementService();
await achievementService.checkAchievements(
userId,
AchievementCategory.Suggestion,
request
);
reply.send(suggestion);
} catch (error) {
return reply.badRequest(
@@ -124,22 +115,14 @@ export default async function (app: FastifyInstance): Promise<void> {
const suggestion = await SuggestionService.acceptSuggestion(id);
await AuditService.logFromRequest(request, {
action: AuditAction.entryUpdate,
category: AuditCategory.admin,
action: AuditAction.ENTRY_UPDATE,
category: AuditCategory.ADMIN,
resourceType: "Suggestion",
resourceId: suggestion.id,
details: `Accepted ${suggestion.entityType} suggestion: ${suggestion.title}`,
success: true,
});
// Check for suggestion achievements for the user who made the suggestion
const achievementService = new AchievementService();
await achievementService.checkAchievements(
suggestion.userId,
AchievementCategory.Suggestion,
request
);
reply.send(suggestion);
} catch (error) {
return reply.badRequest(
@@ -163,22 +146,14 @@ export default async function (app: FastifyInstance): Promise<void> {
const suggestion = await SuggestionService.acceptSuggestionWithEdits(id, editedData);
await AuditService.logFromRequest(request, {
action: AuditAction.entryUpdate,
category: AuditCategory.admin,
action: AuditAction.ENTRY_UPDATE,
category: AuditCategory.ADMIN,
resourceType: "Suggestion",
resourceId: suggestion.id,
details: `Accepted ${suggestion.entityType} suggestion with edits: ${suggestion.title}`,
success: true,
});
// Check for suggestion achievements for the user who made the suggestion
const achievementService = new AchievementService();
await achievementService.checkAchievements(
suggestion.userId,
AchievementCategory.Suggestion,
request
);
reply.send(suggestion);
} catch (error) {
return reply.badRequest(
@@ -202,8 +177,8 @@ export default async function (app: FastifyInstance): Promise<void> {
const suggestion = await SuggestionService.declineSuggestion(id, reason);
await AuditService.logFromRequest(request, {
action: AuditAction.entryUpdate,
category: AuditCategory.admin,
action: AuditAction.ENTRY_UPDATE,
category: AuditCategory.ADMIN,
resourceType: "Suggestion",
resourceId: suggestion.id,
details: `Declined ${suggestion.entityType} suggestion: ${suggestion.title}${reason ? ` (Reason: ${reason})` : ""}`,
@@ -234,8 +209,8 @@ export default async function (app: FastifyInstance): Promise<void> {
const suggestion = await SuggestionService.deleteSuggestion(id, userId, isAdmin);
await AuditService.logFromRequest(request, {
action: AuditAction.entryDelete,
category: isAdmin ? AuditCategory.admin : AuditCategory.content,
action: AuditAction.ENTRY_DELETE,
category: isAdmin ? AuditCategory.ADMIN : AuditCategory.CONTENT,
resourceType: "Suggestion",
resourceId: suggestion.id,
details: `Deleted ${suggestion.entityType} suggestion: ${suggestion.title}`,
+5 -210
View File
@@ -5,56 +5,11 @@
*/
import { FastifyPluginAsync } from "fastify";
import { User, AuditAction, AuditCategory, PrimaryBadge } from "@library/shared-types";
import { User, AuditAction, AuditCategory } from "@library/shared-types";
import { UserService } from "../../services/user.service";
import { AuditService } from "../../services/audit.service";
import { adminGuard } from "../../middleware/admin-guard";
interface UpdateUserSettingsBody {
slug?: string;
displayName?: string;
bio?: string;
profilePublic?: boolean;
primaryBadge?: PrimaryBadge;
website?: string;
discordServer?: string;
bluesky?: string;
github?: string;
linkedin?: string;
twitch?: string;
youtube?: string;
}
interface UserProfileResponse {
id: string;
username: string;
displayName?: string;
avatar?: string;
bio?: string;
slug?: string;
primaryBadge?: PrimaryBadge;
website?: string;
discordServer?: string;
bluesky?: string;
github?: string;
linkedin?: string;
twitch?: string;
youtube?: string;
badges: {
isStaff: boolean;
isMod: boolean;
isVip: boolean;
inDiscord: boolean;
};
stats: {
suggestionsCount: number;
suggestionsAcceptedCount: number;
likesCount: number;
commentsCount: number;
};
createdAt: Date;
}
const usersRoutes: FastifyPluginAsync = async (app) => {
const userService = new UserService();
@@ -68,108 +23,6 @@ const usersRoutes: FastifyPluginAsync = async (app) => {
}
);
app.get<{ Reply: User }>(
"/me",
{
preValidation: [app.authenticate],
},
async (request) => {
const currentUser = request.user as { id: string };
const user = await userService.getUserById(currentUser.id);
if (!user) {
throw new Error("User not found");
}
return user;
}
);
app.put<{ Body: UpdateUserSettingsBody; Reply: User | { error: string } }>(
"/me",
{
preValidation: [app.authenticate],
preHandler: [app.csrfProtection],
},
async (request, reply) => {
const currentUser = request.user as { id: string };
const updates = request.body;
// If slug is being updated, check if it's unique
if (updates.slug) {
const existingUser = await userService.getUserBySlug(updates.slug);
if (existingUser && existingUser.id !== currentUser.id) {
return reply.code(400).send({ error: "Slug already taken" });
}
}
const updatedUser = await userService.updateUserSettings(
currentUser.id,
updates
);
if (!updatedUser) {
return reply.code(404).send({ error: "User not found" });
}
return updatedUser;
}
);
app.get<{
Params: { identifier: string };
Reply: UserProfileResponse | { error: string };
}>(
"/profile/:identifier",
async (request, reply) => {
const { identifier } = request.params;
try {
const profile = await userService.getUserProfile(identifier);
if (!profile) {
return reply.code(404).send({ error: "User not found" });
}
if (!profile.profilePublic) {
// Check if the requesting user is viewing their own profile
const currentUser = request.user as { id: string } | undefined;
if (!currentUser || currentUser.id !== profile.id) {
return reply
.code(403)
.send({ error: "This profile is private" });
}
}
return {
id: profile.id,
username: profile.username,
displayName: profile.displayName,
avatar: profile.avatar,
bio: profile.bio,
slug: profile.slug,
primaryBadge: profile.primaryBadge,
website: profile.website,
discordServer: profile.discordServer,
bluesky: profile.bluesky,
github: profile.github,
linkedin: profile.linkedin,
twitch: profile.twitch,
youtube: profile.youtube,
badges: {
isStaff: profile.isStaff,
isMod: profile.isMod,
isVip: profile.isVip,
inDiscord: profile.inDiscord,
},
stats: profile.stats,
createdAt: profile.createdAt,
};
} catch (error) {
app.log.error({ err: error }, "Error fetching profile");
return reply.code(500).send({ error: "Failed to fetch profile" });
}
}
);
app.get<{ Params: { id: string }; Reply: User | null }>(
"/:id",
{
@@ -201,8 +54,8 @@ const usersRoutes: FastifyPluginAsync = async (app) => {
}
await AuditService.logFromRequest(request, {
action: AuditAction.userBan,
category: AuditCategory.admin,
action: AuditAction.USER_BAN,
category: AuditCategory.ADMIN,
targetUserId: id,
details: `Banned user: ${user.username}`,
});
@@ -225,8 +78,8 @@ const usersRoutes: FastifyPluginAsync = async (app) => {
}
await AuditService.logFromRequest(request, {
action: AuditAction.userUnban,
category: AuditCategory.admin,
action: AuditAction.USER_UNBAN,
category: AuditCategory.ADMIN,
targetUserId: id,
details: `Unbanned user: ${user.username}`,
});
@@ -234,64 +87,6 @@ const usersRoutes: FastifyPluginAsync = async (app) => {
return user;
}
);
app.post<{ Params: { id: string }; Reply: User | { error: string } }>(
"/:id/make-private",
{
preValidation: [app.authenticate, adminGuard],
preHandler: [app.csrfProtection],
},
async (request, reply) => {
const { id } = request.params;
const user = await userService.updateUserSettings(id, { profilePublic: false });
if (!user) {
return reply.code(404).send({ error: "User not found" });
}
await AuditService.logFromRequest(request, {
action: AuditAction.entryUpdate,
category: AuditCategory.admin,
targetUserId: id,
details: `Admin made profile private for user: ${user.username}`,
});
return user;
}
);
app.put<{ Params: { id: string }; Body: UpdateUserSettingsBody; Reply: User | { error: string } }>(
"/:id",
{
preValidation: [app.authenticate, adminGuard],
preHandler: [app.csrfProtection],
},
async (request, reply) => {
const { id } = request.params;
const updates = request.body;
// If slug is being updated, check if it's unique
if (updates.slug) {
const existingUser = await userService.getUserBySlug(updates.slug);
if (existingUser && existingUser.id !== id) {
return reply.code(400).send({ error: "Slug already taken" });
}
}
const updatedUser = await userService.updateUserSettings(id, updates);
if (!updatedUser) {
return reply.code(404).send({ error: "User not found" });
}
await AuditService.logFromRequest(request, {
action: AuditAction.entryUpdate,
category: AuditCategory.admin,
targetUserId: id,
details: `Admin updated profile for user: ${updatedUser.username}`,
});
return updatedUser;
}
);
};
export default usersRoutes;
-772
View File
@@ -1,772 +0,0 @@
/**
* @copyright 2026 NHCarrigan
* @license Naomi's Public License
* @author Naomi Carrigan
*/
import type { FastifyRequest } from "fastify";
import {
ACHIEVEMENTS,
ACHIEVEMENT_LIST,
AchievementCategory,
AchievementDefinition,
AchievementProgress,
AuditAction,
AuditCategory,
UserAchievementSummary,
} from "@library/shared-types";
import { prisma } from "../lib/prisma";
import { AuditService } from "./audit.service";
export class AchievementService {
/**
* Check and award achievements for a user after an action.
* Returns list of newly earned achievements.
*/
async checkAchievements(
userId: string,
category: AchievementCategory,
req: FastifyRequest,
): Promise<AchievementDefinition[]> {
const relevantAchievements = ACHIEVEMENT_LIST.filter(
(ach) => ach.category === category,
);
const newlyEarned: AchievementDefinition[] = [];
for (const achievement of relevantAchievements) {
const userAchievement = await prisma.userAchievement.findUnique({
where: {
userId_achievementKey: {
userId,
achievementKey: achievement.key,
},
},
});
// Skip already earned achievements
if (userAchievement?.earned) {
continue;
}
// Check if achievement is now earned
const earned = await this.checkAchievementCondition(userId, achievement);
if (earned) {
await this.awardAchievement(userId, achievement, req);
newlyEarned.push(achievement);
}
}
return newlyEarned;
}
/**
* Award an achievement to a user.
*/
private async awardAchievement(
userId: string,
achievement: AchievementDefinition,
req: FastifyRequest,
): Promise<void> {
await prisma.userAchievement.upsert({
where: {
userId_achievementKey: {
userId,
achievementKey: achievement.key,
},
},
create: {
userId,
achievementKey: achievement.key,
progress: 100,
earned: true,
earnedAt: new Date(),
},
update: {
earned: true,
earnedAt: new Date(),
progress: 100,
},
});
// Update user's achievement points
await prisma.user.update({
where: { id: userId },
data: {
achievementPoints: {
increment: achievement.points,
},
},
});
// Log the achievement unlock
await AuditService.logFromRequest(req, {
action: AuditAction.achievementUnlocked,
category: AuditCategory.content,
details: `Unlocked achievement: ${achievement.title}`,
});
}
/**
* Check if a specific achievement condition is met.
*/
private async checkAchievementCondition(
userId: string,
achievement: AchievementDefinition,
): Promise<boolean> {
switch (achievement.category) {
case AchievementCategory.Suggestion:
return await this.checkSuggestionAchievement(userId, achievement);
case AchievementCategory.Like:
return await this.checkLikeAchievement(userId, achievement);
case AchievementCategory.Comment:
return await this.checkCommentAchievement(userId, achievement);
case AchievementCategory.Engagement:
return await this.checkEngagementAchievement(userId, achievement);
case AchievementCategory.Report:
return await this.checkReportAchievement(userId, achievement);
default:
return false;
}
}
/**
* Check suggestion-based achievements.
*/
private async checkSuggestionAchievement(
userId: string,
achievement: AchievementDefinition,
): Promise<boolean> {
const { requirements } = achievement;
// Count-based achievements (total suggestions)
if (
achievement.key.startsWith("suggestion_first_steps") ||
achievement.key.startsWith("suggestion_contributor") ||
achievement.key.startsWith("suggestion_dedicated") ||
achievement.key.startsWith("suggestion_master") ||
achievement.key.startsWith("suggestion_legend")
) {
const count = await prisma.suggestion.count({
where: { userId },
});
return count >= (requirements.count ?? 0);
}
// Accepted suggestion achievements
if (achievement.key.startsWith("suggestion_quality")) {
const count = await prisma.suggestion.count({
where: {
userId,
status: "ACCEPTED",
},
});
return count >= (requirements.count ?? 0);
}
// Approved achievement (first accepted)
if (achievement.key === "suggestion_approved") {
const count = await prisma.suggestion.count({
where: {
userId,
status: "ACCEPTED",
},
});
return count >= 1;
}
// Acceptance rate achievements
if (achievement.key.startsWith("suggestion_acceptance")) {
const total = await prisma.suggestion.count({
where: { userId },
});
const accepted = await prisma.suggestion.count({
where: {
userId,
status: "ACCEPTED",
},
});
if (total < (requirements.count ?? 0)) {
return false;
}
const rate = accepted / total;
return rate >= (requirements.rate ?? 0);
}
// Diversity achievement (all media types)
if (achievement.key === "suggestion_renaissance") {
const types = await prisma.suggestion.groupBy({
by: ["entityType"],
where: {
userId,
status: "ACCEPTED",
},
});
return types.length >= 6;
}
// Enthusiast (5 in one day)
if (achievement.key === "suggestion_enthusiast") {
const today = new Date();
today.setHours(0, 0, 0, 0);
const tomorrow = new Date(today);
tomorrow.setDate(tomorrow.getDate() + 1);
const count = await prisma.suggestion.count({
where: {
userId,
createdAt: {
gte: today,
lt: tomorrow,
},
},
});
return count >= 5;
}
return false;
}
/**
* Check like-based achievements.
*/
private async checkLikeAchievement(
userId: string,
achievement: AchievementDefinition,
): Promise<boolean> {
const { requirements } = achievement;
// Count-based achievements (total likes)
if (
achievement.key.startsWith("like_first") ||
achievement.key.startsWith("like_enthusiast") ||
achievement.key.startsWith("like_fan") ||
achievement.key.startsWith("like_super") ||
achievement.key.startsWith("like_mega") ||
achievement.key.startsWith("like_legendary")
) {
const count = await prisma.like.count({
where: { userId },
});
return count >= (requirements.count ?? 0);
}
// Media-specific achievements
if (achievement.key === "like_book_lover") {
const count = await prisma.like.count({
where: {
userId,
entityType: "book",
},
});
return count >= 50;
}
if (achievement.key === "like_gamer") {
const count = await prisma.like.count({
where: {
userId,
entityType: "game",
},
});
return count >= 50;
}
if (achievement.key === "like_cinephile") {
const count = await prisma.like.count({
where: {
userId,
entityType: "show",
},
});
return count >= 50;
}
if (achievement.key === "like_music") {
const count = await prisma.like.count({
where: {
userId,
entityType: "music",
},
});
return count >= 50;
}
// Diversity achievement
if (achievement.key === "like_diverse") {
const types = await prisma.like.groupBy({
by: ["entityType"],
where: { userId },
});
return types.length >= 6;
}
// Binge liker (20+ in one day)
if (achievement.key === "like_binge") {
const today = new Date();
today.setHours(0, 0, 0, 0);
const tomorrow = new Date(today);
tomorrow.setDate(tomorrow.getDate() + 1);
const count = await prisma.like.count({
where: {
userId,
createdAt: {
gte: today,
lt: tomorrow,
},
},
});
return count >= 20;
}
return false;
}
/**
* Check comment-based achievements.
*/
private async checkCommentAchievement(
userId: string,
achievement: AchievementDefinition,
): Promise<boolean> {
const { requirements } = achievement;
// Count-based achievements (total comments)
if (
achievement.key.startsWith("comment_first") ||
achievement.key.startsWith("comment_reviewer") ||
achievement.key.startsWith("comment_critic") ||
achievement.key.startsWith("comment_expert") ||
achievement.key.startsWith("comment_master") ||
achievement.key.startsWith("comment_legend")
) {
const count = await prisma.comment.count({
where: { userId },
});
return count >= (requirements.count ?? 0);
}
// Length-based achievements
if (achievement.key === "comment_detailed") {
const count = await prisma.comment.count({
where: {
userId,
content: {
// MongoDB doesn't have a direct length check, so we'll fetch and check
},
},
});
// Fetch all comments and check length
const comments = await prisma.comment.findMany({
where: { userId },
select: { content: true },
});
const longComments = comments.filter((c) => c.content.length >= 500);
return longComments.length >= 10;
}
if (achievement.key === "comment_essay") {
const comments = await prisma.comment.findMany({
where: { userId },
select: { content: true },
});
const longComments = comments.filter((c) => c.content.length >= 1000);
return longComments.length >= 5;
}
if (achievement.key === "comment_novel") {
const comments = await prisma.comment.findMany({
where: { userId },
select: { content: true },
});
const longComments = comments.filter((c) => c.content.length >= 2000);
return longComments.length >= 3;
}
// Thoughtful reviewer (50 different items)
if (achievement.key === "comment_thoughtful") {
// Count unique entity IDs
const comments = await prisma.comment.findMany({
where: { userId },
select: {
bookId: true,
gameId: true,
showId: true,
mangaId: true,
musicId: true,
artId: true,
},
});
const uniqueItems = new Set<string>();
comments.forEach((c) => {
const entityId =
c.bookId ??
c.gameId ??
c.showId ??
c.mangaId ??
c.musicId ??
c.artId;
if (entityId) {
uniqueItems.add(entityId);
}
});
return uniqueItems.size >= 50;
}
// Diversity achievement
if (achievement.key === "comment_diverse") {
const comments = await prisma.comment.findMany({
where: { userId },
select: {
bookId: true,
gameId: true,
showId: true,
mangaId: true,
musicId: true,
artId: true,
},
});
const types = new Set<string>();
comments.forEach((c) => {
if (c.bookId) {
types.add("book");
}
if (c.gameId) {
types.add("game");
}
if (c.showId) {
types.add("show");
}
if (c.mangaId) {
types.add("manga");
}
if (c.musicId) {
types.add("music");
}
if (c.artId) {
types.add("art");
}
});
return types.size >= 6;
}
return false;
}
/**
* Check engagement-based achievements.
*/
private async checkEngagementAchievement(
userId: string,
achievement: AchievementDefinition,
): Promise<boolean> {
const user = await prisma.user.findUnique({
where: { id: userId },
select: {
currentStreak: true,
createdAt: true,
},
});
if (!user) {
return false;
}
// Welcome achievement
if (achievement.key === "engagement_welcome") {
return true; // Awarded on first login
}
// Streak-based achievements
if (achievement.key.startsWith("engagement_streak")) {
return user.currentStreak >= (achievement.requirements.streak ?? 0);
}
// Triple threat (suggestion, like, comment in same day)
if (achievement.key === "engagement_triple_threat") {
const today = new Date();
today.setHours(0, 0, 0, 0);
const tomorrow = new Date(today);
tomorrow.setDate(tomorrow.getDate() + 1);
const [hasSuggestion, hasLike, hasComment] = await Promise.all([
prisma.suggestion.count({
where: {
userId,
createdAt: { gte: today, lt: tomorrow },
},
}),
prisma.like.count({
where: {
userId,
createdAt: { gte: today, lt: tomorrow },
},
}),
prisma.comment.count({
where: {
userId,
createdAt: { gte: today, lt: tomorrow },
},
}),
]);
return hasSuggestion > 0 && hasLike > 0 && hasComment > 0;
}
// Power user (triple threat 10 times) - would need special tracking
// Early adopter achievements
if (
achievement.key === "engagement_early_adopter" ||
achievement.key === "engagement_founding_100" ||
achievement.key === "engagement_founding_1000"
) {
const userRank = await prisma.user.count({
where: {
createdAt: {
lt: user.createdAt,
},
},
});
if (achievement.key === "engagement_early_adopter") {
return userRank < 10;
}
if (achievement.key === "engagement_founding_100") {
return userRank < 100;
}
if (achievement.key === "engagement_founding_1000") {
return userRank < 1000;
}
}
// Account age achievements
if (achievement.key.startsWith("engagement_veteran")) {
const daysSinceCreation = Math.floor(
(Date.now() - user.createdAt.getTime()) / (1000 * 60 * 60 * 24),
);
return daysSinceCreation >= (achievement.requirements.dayRange ?? 0);
}
return false;
}
/**
* Check report-based achievements.
*/
private async checkReportAchievement(
userId: string,
achievement: AchievementDefinition,
): Promise<boolean> {
const { requirements } = achievement;
// Count ACTION_TAKEN reports
if (
achievement.key.startsWith("report_watchful") ||
achievement.key.startsWith("report_guardian") ||
achievement.key.startsWith("report_protector") ||
achievement.key.startsWith("report_vigilant")
) {
const profileReports = await prisma.profileReport.count({
where: {
reporterId: userId,
status: "ACTION_TAKEN",
},
});
const commentReports = await prisma.commentReport.count({
where: {
reporterId: userId,
status: "ACTION_TAKEN",
},
});
return profileReports + commentReports >= (requirements.count ?? 0);
}
// Accuracy achievements
if (achievement.key.startsWith("report_accuracy")) {
const totalProfileReports = await prisma.profileReport.count({
where: {
reporterId: userId,
status: {
not: "PENDING",
},
},
});
const totalCommentReports = await prisma.commentReport.count({
where: {
reporterId: userId,
status: {
not: "PENDING",
},
},
});
const total = totalProfileReports + totalCommentReports;
if (total < (requirements.count ?? 10)) {
return false;
}
const actionTakenProfile = await prisma.profileReport.count({
where: {
reporterId: userId,
status: "ACTION_TAKEN",
},
});
const actionTakenComment = await prisma.commentReport.count({
where: {
reporterId: userId,
status: "ACTION_TAKEN",
},
});
const actionTaken = actionTakenProfile + actionTakenComment;
const rate = actionTaken / total;
return rate >= (requirements.rate ?? 0);
}
// Volume achievement (total reports)
if (achievement.key === "report_volume") {
const profileReports = await prisma.profileReport.count({
where: { reporterId: userId },
});
const commentReports = await prisma.commentReport.count({
where: { reporterId: userId },
});
return profileReports + commentReports >= 50;
}
return false;
}
/**
* Get a user's achievement progress and summary.
*/
async getUserAchievementSummary(
userId: string,
): Promise<UserAchievementSummary> {
const userAchievements = await prisma.userAchievement.findMany({
where: { userId },
orderBy: { earnedAt: "desc" },
});
const earnedAchievements = userAchievements.filter((ua) => ua.earned);
const totalPoints = earnedAchievements.reduce((sum, ua) => {
const definition = ACHIEVEMENTS[ua.achievementKey];
return sum + (definition?.points ?? 0);
}, 0);
const recentAchievements: AchievementProgress[] = earnedAchievements
.slice(0, 5)
.map((ua) => ({
definition: ACHIEVEMENTS[ua.achievementKey],
progress: ua.progress,
earned: ua.earned,
earnedAt: ua.earnedAt ?? undefined,
}));
// Progress by category
const progressByCategory = Object.values(AchievementCategory).map(
(category) => {
const categoryAchievements = ACHIEVEMENT_LIST.filter(
(a) => a.category === category,
);
const earned = earnedAchievements.filter(
(ua) => ACHIEVEMENTS[ua.achievementKey]?.category === category,
).length;
return {
category,
earned,
total: categoryAchievements.length,
};
},
);
return {
totalPoints,
totalEarned: earnedAchievements.length,
recentAchievements,
progressByCategory,
};
}
/**
* Get all achievements with progress for a user.
*/
async getUserAchievementProgress(
userId: string,
): Promise<AchievementProgress[]> {
const userAchievements = await prisma.userAchievement.findMany({
where: { userId },
});
const progressMap = new Map(
userAchievements.map((ua) => [ua.achievementKey, ua]),
);
return ACHIEVEMENT_LIST.map((definition) => {
const userAchievement = progressMap.get(definition.key);
return {
definition,
progress: userAchievement?.progress ?? 0,
earned: userAchievement?.earned ?? false,
earnedAt: userAchievement?.earnedAt ?? undefined,
};
});
}
/**
* Update login streak for a user.
*/
async updateLoginStreak(userId: string): Promise<void> {
const user = await prisma.user.findUnique({
where: { id: userId },
select: {
currentStreak: true,
lastStreakCheck: true,
},
});
if (!user) {
return;
}
const now = new Date();
const yesterday = new Date(now);
yesterday.setDate(yesterday.getDate() - 1);
yesterday.setHours(0, 0, 0, 0);
const lastCheck = user.lastStreakCheck;
const isConsecutive =
lastCheck &&
lastCheck >= yesterday &&
lastCheck < new Date(now.setHours(0, 0, 0, 0));
await prisma.user.update({
where: { id: userId },
data: {
currentStreak: isConsecutive ? user.currentStreak + 1 : 1,
lastStreakCheck: new Date(),
},
});
}
}
-353
View File
@@ -1,353 +0,0 @@
/**
* @copyright 2026 NHCarrigan
* @license Naomi's Public License
* @author Naomi Carrigan
*/
import type {
Activity,
ActivityFeedResponse,
ActivityUser,
SuggestionActivity,
LikeActivity,
CommentActivity,
AchievementActivity,
} from "@library/shared-types";
import { ACHIEVEMENTS, ActivityType } from "@library/shared-types";
import { prisma } from "../lib/prisma";
export class ActivityService {
private prisma = prisma;
constructor() {}
/**
* Get activity feed with pagination.
*/
async getActivityFeed(
limit = 50,
offset = 0,
userId?: string
): Promise<ActivityFeedResponse> {
// Fetch suggestions, likes, comments, and achievements
const [suggestions, likes, comments, achievements] = await Promise.all([
this.getSuggestionActivities(limit, offset, userId),
this.getLikeActivities(limit, offset, userId),
this.getCommentActivities(limit, offset, userId),
this.getAchievementActivities(limit, offset, userId),
]);
// Combine and sort by createdAt
const activities: Activity[] = [
...suggestions,
...likes,
...comments,
...achievements,
].sort((a, b) => {
return new Date(b.createdAt).getTime() - new Date(a.createdAt).getTime();
});
// Apply pagination to combined results
const paginatedActivities = activities.slice(offset, offset + limit);
const hasMore = activities.length > offset + limit;
return {
activities: paginatedActivities,
total: activities.length,
hasMore,
};
}
/**
* Get suggestion activities.
*/
private async getSuggestionActivities(
limit: number,
offset: number,
userId?: string
): Promise<SuggestionActivity[]> {
const where = userId
? { userId, user: { profilePublic: true, isBanned: false } }
: { user: { profilePublic: true, isBanned: false } };
const suggestions = await this.prisma.suggestion.findMany({
where,
include: {
user: {
select: {
id: true,
username: true,
slug: true,
avatar: true,
primaryBadge: true,
isVip: true,
isMod: true,
isStaff: true,
},
},
},
orderBy: { createdAt: "desc" },
take: limit * 2, // Get more since we're combining
});
return suggestions.map((suggestion) => ({
id: `suggestion-${suggestion.id}`,
type: ActivityType.suggestion,
user: suggestion.user as ActivityUser,
entityType: suggestion.entityType,
suggestionTitle: suggestion.title,
status: suggestion.status,
createdAt: suggestion.createdAt,
}));
}
/**
* Get like activities.
*/
private async getLikeActivities(
limit: number,
offset: number,
userId?: string
): Promise<LikeActivity[]> {
const where = userId
? { userId, user: { profilePublic: true, isBanned: false } }
: { user: { profilePublic: true, isBanned: false } };
const likes = await this.prisma.like.findMany({
where,
include: {
user: {
select: {
id: true,
username: true,
slug: true,
avatar: true,
primaryBadge: true,
isVip: true,
isMod: true,
isStaff: true,
},
},
},
orderBy: { createdAt: "desc" },
take: limit * 2,
});
// For each like, fetch the entity title
const likesWithTitles = await Promise.all(
likes.map(async (like): Promise<LikeActivity> => {
const entityTitle = await this.getEntityTitle(
like.entityType,
like.entityId
);
return {
id: `like-${like.id}`,
type: ActivityType.like,
user: like.user as ActivityUser,
entityType: like.entityType,
entityId: like.entityId,
entityTitle,
createdAt: like.createdAt,
};
})
);
return likesWithTitles;
}
/**
* Get comment activities.
*/
private async getCommentActivities(
limit: number,
offset: number,
userId?: string
): Promise<CommentActivity[]> {
const where = userId
? { userId, user: { profilePublic: true, isBanned: false } }
: { user: { profilePublic: true, isBanned: false } };
const comments = await this.prisma.comment.findMany({
where,
include: {
user: {
select: {
id: true,
username: true,
slug: true,
avatar: true,
primaryBadge: true,
isVip: true,
isMod: true,
isStaff: true,
},
},
game: { select: { id: true, title: true } },
book: { select: { id: true, title: true } },
music: { select: { id: true, title: true } },
art: { select: { id: true, title: true } },
show: { select: { id: true, title: true } },
manga: { select: { id: true, title: true } },
},
orderBy: { createdAt: "desc" },
take: limit * 2,
});
return comments.map((comment) => {
let entityType = "";
let entityId = "";
let entityTitle = "";
if (comment.game) {
entityType = "game";
entityId = comment.game.id;
entityTitle = comment.game.title;
} else if (comment.book) {
entityType = "book";
entityId = comment.book.id;
entityTitle = comment.book.title;
} else if (comment.music) {
entityType = "music";
entityId = comment.music.id;
entityTitle = comment.music.title;
} else if (comment.art) {
entityType = "art";
entityId = comment.art.id;
entityTitle = comment.art.title;
} else if (comment.show) {
entityType = "show";
entityId = comment.show.id;
entityTitle = comment.show.title;
} else if (comment.manga) {
entityType = "manga";
entityId = comment.manga.id;
entityTitle = comment.manga.title;
}
// Get first 100 characters of comment
const commentPreview =
comment.content.length > 100
? `${comment.content.slice(0, 100)}...`
: comment.content;
return {
id: `comment-${comment.id}`,
type: ActivityType.comment,
user: comment.user as ActivityUser,
entityType,
entityId,
entityTitle,
commentPreview,
createdAt: comment.createdAt,
};
});
}
/**
* Get achievement activities.
*/
private async getAchievementActivities(
limit: number,
offset: number,
userId?: string
): Promise<AchievementActivity[]> {
const where = userId
? {
userId,
earned: true,
user: { profilePublic: true, isBanned: false },
}
: { earned: true, user: { profilePublic: true, isBanned: false } };
const userAchievements = await this.prisma.userAchievement.findMany({
where,
include: {
user: {
select: {
id: true,
username: true,
slug: true,
avatar: true,
primaryBadge: true,
isVip: true,
isMod: true,
isStaff: true,
},
},
},
orderBy: { earnedAt: "desc" },
take: limit * 2,
});
return userAchievements
.filter((ua) => ua.earnedAt) // Only show earned achievements
.map((ua) => {
const achievement = ACHIEVEMENTS[ua.achievementKey];
return {
id: `achievement-${ua.id}`,
type: ActivityType.achievement,
user: ua.user as ActivityUser,
achievementKey: ua.achievementKey,
achievementName: achievement.title,
achievementIcon: achievement.icon,
achievementPoints: achievement.points,
createdAt: ua.earnedAt!,
};
});
}
/**
* Helper to get entity title by type and ID.
*/
private async getEntityTitle(
entityType: string,
entityId: string
): Promise<string> {
switch (entityType) {
case "game": {
const game = await this.prisma.game.findUnique({
where: { id: entityId },
select: { title: true },
});
return game?.title || "Unknown Game";
}
case "book": {
const book = await this.prisma.book.findUnique({
where: { id: entityId },
select: { title: true },
});
return book?.title || "Unknown Book";
}
case "music": {
const music = await this.prisma.music.findUnique({
where: { id: entityId },
select: { title: true },
});
return music?.title || "Unknown Music";
}
case "art": {
const art = await this.prisma.art.findUnique({
where: { id: entityId },
select: { title: true },
});
return art?.title || "Unknown Art";
}
case "show": {
const show = await this.prisma.show.findUnique({
where: { id: entityId },
select: { title: true },
});
return show?.title || "Unknown Show";
}
case "manga": {
const manga = await this.prisma.manga.findUnique({
where: { id: entityId },
select: { title: true },
});
return manga?.title || "Unknown Manga";
}
default:
return "Unknown Item";
}
}
}
-62
View File
@@ -6,68 +6,12 @@
import { Art, CreateArtDto, UpdateArtDto } from "@library/shared-types";
import { prisma } from "../lib/prisma";
import {
validateUrl,
validateStringLength,
MAX_LENGTHS,
} from "../utils/validation";
export class ArtService {
private prisma = prisma;
constructor() {}
/**
* Validate art data for security.
*/
private validateArtData(data: CreateArtDto | UpdateArtDto): void {
// Validate string lengths
if (!validateStringLength(data.title, MAX_LENGTHS.TITLE)) {
throw new Error(`Title must be ${MAX_LENGTHS.TITLE} characters or less.`);
}
if (!validateStringLength(data.artist, MAX_LENGTHS.AUTHOR)) {
throw new Error(`Artist must be ${MAX_LENGTHS.AUTHOR} characters or less.`);
}
if (!validateStringLength(data.description, MAX_LENGTHS.DESCRIPTION)) {
throw new Error(`Description must be ${MAX_LENGTHS.DESCRIPTION} characters or less.`);
}
// Validate image URL (required)
if (!data.imageUrl) {
throw new Error("Image URL is required.");
}
if (!validateUrl(data.imageUrl)) {
throw new Error("Invalid image URL. Only http and https URLs are allowed.");
}
if (!validateStringLength(data.imageUrl, MAX_LENGTHS.URL)) {
throw new Error(`Image URL must be ${MAX_LENGTHS.URL} characters or less.`);
}
// Validate tags
if (data.tags) {
for (const tag of data.tags) {
if (!validateStringLength(tag, MAX_LENGTHS.TAGS)) {
throw new Error(`Each tag must be ${MAX_LENGTHS.TAGS} characters or less.`);
}
}
}
// Validate link URLs
if (data.links) {
for (const link of data.links) {
if (!validateUrl(link.url)) {
throw new Error(`Invalid link URL: ${link.title}. Only http and https URLs are allowed.`);
}
if (!validateStringLength(link.title, MAX_LENGTHS.AUTHOR)) {
throw new Error(`Link title must be ${MAX_LENGTHS.AUTHOR} characters or less.`);
}
if (!validateStringLength(link.url, MAX_LENGTHS.URL)) {
throw new Error(`Link URL must be ${MAX_LENGTHS.URL} characters or less.`);
}
}
}
}
/**
* Get all art pieces.
*/
@@ -112,9 +56,6 @@ export class ArtService {
* Create new art piece.
*/
async createArt(data: CreateArtDto): Promise<Art> {
// Validate input
this.validateArtData(data);
const art = await this.prisma.art.create({
data,
});
@@ -134,9 +75,6 @@ export class ArtService {
* Update art by ID.
*/
async updateArt(id: string, data: UpdateArtDto): Promise<Art> {
// Validate input
this.validateArtData(data);
const art = await this.prisma.art.update({
where: { id },
data,
+1 -1
View File
@@ -36,7 +36,7 @@ export const AuditService = {
request: FastifyRequest,
data: Omit<AuditLogData, "userId">
) {
const userId = ((request as any).user as { id?: string } | undefined)?.id;
const userId = (request.user as { id?: string } | undefined)?.id;
return this.log(
{
-27
View File
@@ -71,15 +71,6 @@ export class AuthService {
username: dbUser.username,
email: dbUser.email,
avatar: dbUser.avatar || undefined,
slug: dbUser.slug || undefined,
displayName: dbUser.displayName || undefined,
bio: dbUser.bio || undefined,
profilePublic: dbUser.profilePublic,
website: dbUser.website || undefined,
discordServer: dbUser.discordServer || undefined,
bluesky: dbUser.bluesky || undefined,
github: dbUser.github || undefined,
linkedin: dbUser.linkedin || undefined,
isAdmin: dbUser.isAdmin,
isBanned: dbUser.isBanned,
inDiscord: dbUser.inDiscord,
@@ -176,15 +167,6 @@ export class AuthService {
username: dbUser.username,
email: dbUser.email,
avatar: dbUser.avatar || undefined,
slug: dbUser.slug || undefined,
displayName: dbUser.displayName || undefined,
bio: dbUser.bio || undefined,
profilePublic: dbUser.profilePublic,
website: dbUser.website || undefined,
discordServer: dbUser.discordServer || undefined,
bluesky: dbUser.bluesky || undefined,
github: dbUser.github || undefined,
linkedin: dbUser.linkedin || undefined,
isAdmin: dbUser.isAdmin,
isBanned: dbUser.isBanned,
inDiscord: dbUser.inDiscord,
@@ -235,15 +217,6 @@ export class AuthService {
username: dbUser.username,
email: dbUser.email,
avatar: dbUser.avatar || undefined,
slug: dbUser.slug || undefined,
displayName: dbUser.displayName || undefined,
bio: dbUser.bio || undefined,
profilePublic: dbUser.profilePublic,
website: dbUser.website || undefined,
discordServer: dbUser.discordServer || undefined,
bluesky: dbUser.bluesky || undefined,
github: dbUser.github || undefined,
linkedin: dbUser.linkedin || undefined,
isAdmin: dbUser.isAdmin,
isBanned: dbUser.isBanned,
inDiscord: dbUser.inDiscord,
-109
View File
@@ -6,89 +6,12 @@
import { Book, BookStatus, CreateBookDto, UpdateBookDto } from "@library/shared-types";
import { prisma } from "../lib/prisma";
import {
validateUrl,
validateRating,
validateStringLength,
validateDataUrl,
MAX_LENGTHS,
} from "../utils/validation";
export class BookService {
private prisma = prisma;
constructor() {}
/**
* Validate book data for security.
*/
private validateBookData(data: CreateBookDto | UpdateBookDto): void {
// Validate string lengths
if (!validateStringLength(data.title, MAX_LENGTHS.TITLE)) {
throw new Error(`Title must be ${MAX_LENGTHS.TITLE} characters or less.`);
}
if (!validateStringLength(data.author, MAX_LENGTHS.AUTHOR)) {
throw new Error(`Author must be ${MAX_LENGTHS.AUTHOR} characters or less.`);
}
if (!validateStringLength(data.isbn, MAX_LENGTHS.ISBN)) {
throw new Error(`ISBN must be ${MAX_LENGTHS.ISBN} characters or less.`);
}
if (!validateStringLength(data.notes, MAX_LENGTHS.NOTES)) {
throw new Error(`Notes must be ${MAX_LENGTHS.NOTES} characters or less.`);
}
// Validate rating
if (!validateRating(data.rating)) {
throw new Error("Rating must be an integer between 0 and 10.");
}
if (data.coverImage) {
if (data.coverImage.startsWith("data:")) {
const base64Data = data.coverImage.split(",")[1];
if (!base64Data) {
throw new Error("Invalid image data URL format.");
}
const sizeInBytes = base64Data.length * 0.75;
if (sizeInBytes > MAX_LENGTHS.DATA_URL) {
throw new Error("Cover image must be under 5MB.");
}
if (!validateDataUrl(data.coverImage)) {
throw new Error("Invalid image data URL.");
}
} else {
if (!validateStringLength(data.coverImage, MAX_LENGTHS.URL)) {
throw new Error(`Cover image URL must be ${MAX_LENGTHS.URL} characters or less.`);
}
if (!validateUrl(data.coverImage)) {
throw new Error("Invalid cover image URL. Only http and https URLs are allowed.");
}
}
}
// Validate tags
if (data.tags) {
for (const tag of data.tags) {
if (!validateStringLength(tag, MAX_LENGTHS.TAGS)) {
throw new Error(`Each tag must be ${MAX_LENGTHS.TAGS} characters or less.`);
}
}
}
// Validate link URLs
if (data.links) {
for (const link of data.links) {
if (!validateUrl(link.url)) {
throw new Error(`Invalid link URL: ${link.title}. Only http and https URLs are allowed.`);
}
if (!validateStringLength(link.title, MAX_LENGTHS.TITLE)) {
throw new Error(`Link title must be ${MAX_LENGTHS.TITLE} characters or less.`);
}
if (!validateStringLength(link.url, MAX_LENGTHS.URL)) {
throw new Error(`Link URL must be ${MAX_LENGTHS.URL} characters or less.`);
}
}
}
}
/**
* Get all books.
*/
@@ -101,7 +24,6 @@ export class BookService {
...book,
status: book.status as unknown as BookStatus,
dateAdded: book.dateAdded,
dateStarted: book.dateStarted || undefined,
dateFinished: book.dateFinished || undefined,
tags: book.tags ?? [],
links: book.links ?? [],
@@ -124,7 +46,6 @@ export class BookService {
...book,
status: book.status as unknown as BookStatus,
dateAdded: book.dateAdded,
dateStarted: book.dateStarted || undefined,
dateFinished: book.dateFinished || undefined,
tags: book.tags ?? [],
links: book.links ?? [],
@@ -133,35 +54,10 @@ export class BookService {
};
}
/**
* Get all books in a series, ordered by seriesOrder.
*/
async getBooksBySeries(seriesName: string): Promise<Book[]> {
const books = await this.prisma.book.findMany({
where: { series: seriesName },
orderBy: { seriesOrder: "asc" },
});
return books.map((book) => ({
...book,
status: book.status as unknown as BookStatus,
dateAdded: book.dateAdded,
dateStarted: book.dateStarted || undefined,
dateFinished: book.dateFinished || undefined,
tags: book.tags ?? [],
links: book.links ?? [],
createdAt: book.createdAt,
updatedAt: book.updatedAt,
}));
}
/**
* Create new book.
*/
async createBook(data: CreateBookDto): Promise<Book> {
// Validate input
this.validateBookData(data);
const book = await this.prisma.book.create({
data: {
...data,
@@ -173,7 +69,6 @@ export class BookService {
...book,
status: book.status as unknown as BookStatus,
dateAdded: book.dateAdded,
dateStarted: book.dateStarted || undefined,
dateFinished: book.dateFinished || undefined,
tags: book.tags ?? [],
links: book.links ?? [],
@@ -186,9 +81,6 @@ export class BookService {
* Update book by ID.
*/
async updateBook(id: string, data: UpdateBookDto): Promise<Book> {
// Validate input
this.validateBookData(data);
const updateData = { ...data };
if (updateData.status) {
updateData.status = updateData.status.toUpperCase() as any;
@@ -203,7 +95,6 @@ export class BookService {
...book,
status: book.status as unknown as BookStatus,
dateAdded: book.dateAdded,
dateStarted: book.dateStarted || undefined,
dateFinished: book.dateFinished || undefined,
tags: book.tags ?? [],
links: book.links ?? [],
@@ -1,369 +0,0 @@
/**
* @copyright NHCarrigan
* @license Naomi's Public License
* @author Naomi Carrigan
*/
import {
ReportStatus as PrismaReportStatus,
ReportReason as PrismaReportReason,
} from "@prisma/client";
import type {
CreateCommentReportDto,
CommentReportWithDetails,
ReportStatus,
UpdateCommentReportDto,
} from "@library/shared-types";
import { ReportReason } from "@library/shared-types";
import { prisma } from "../lib/prisma.js";
export class CommentReportService {
private prisma = prisma;
/**
* Convert Prisma ReportReason to shared-types ReportReason
*/
private toPrismaReportReason(reason: ReportReason): PrismaReportReason {
return reason as unknown as PrismaReportReason;
}
/**
* Convert Prisma ReportStatus to shared-types ReportStatus
*/
private toPrismaReportStatus(status: ReportStatus): PrismaReportStatus {
return status as unknown as PrismaReportStatus;
}
/**
* Convert Prisma enum back to shared-types enum
*/
private fromPrismaReportReason(reason: PrismaReportReason): ReportReason {
return reason as unknown as ReportReason;
}
/**
* Convert Prisma enum back to shared-types enum
*/
private fromPrismaReportStatus(status: PrismaReportStatus): ReportStatus {
return status as unknown as ReportStatus;
}
/**
* Create a new comment report.
*
* @param reporterId - The ID of the user making the report
* @param createDto - The report details
* @returns The created report
* @throws Error if user already has a pending report for this comment
*/
async createReport(
reporterId: string,
createDto: CreateCommentReportDto,
): Promise<CommentReportWithDetails> {
// Check if user already has a pending report for this comment
const existingReport = await this.prisma.commentReport.findFirst({
where: {
reporterId,
reportedCommentId: createDto.reportedCommentId,
status: PrismaReportStatus.PENDING,
},
});
if (existingReport) {
throw new Error(
"You already have a pending report for this comment. Please wait for it to be reviewed.",
);
}
// Check if user has reached the limit of pending reports (5 max)
const pendingReportsCount = await this.prisma.commentReport.count({
where: {
reporterId,
status: PrismaReportStatus.PENDING,
},
});
if (pendingReportsCount >= 5) {
throw new Error(
"You have reached the maximum number of pending reports (5). Please wait for your existing reports to be reviewed.",
);
}
const report = await this.prisma.commentReport.create({
data: {
reporterId,
reportedCommentId: createDto.reportedCommentId,
reason: this.toPrismaReportReason(createDto.reason),
details: createDto.details,
},
include: {
reportedComment: {
include: {
user: {
select: {
id: true,
username: true,
displayName: true,
avatar: true,
},
},
},
},
reporter: {
select: {
id: true,
username: true,
displayName: true,
avatar: true,
},
},
},
});
return {
id: report.id,
reportedCommentId: report.reportedCommentId,
reporterId: report.reporterId,
reason: this.fromPrismaReportReason(report.reason),
details: report.details,
status: this.fromPrismaReportStatus(report.status),
reviewedBy: report.reviewedBy ?? undefined,
reviewNotes: report.reviewNotes ?? undefined,
createdAt: report.createdAt,
updatedAt: report.updatedAt,
reportedComment: {
id: report.reportedComment.id,
content: report.reportedComment.content,
rawContent: report.reportedComment.rawContent ?? undefined,
userId: report.reportedComment.userId,
user: report.reportedComment.user,
},
reporter: report.reporter,
};
}
/**
* Get all comment reports (admin only). Optionally filter by status.
*
* @param status - Optional status filter
* @returns All reports matching the filter
*/
async getAllReports(
status?: ReportStatus,
): Promise<CommentReportWithDetails[]> {
const reports = await this.prisma.commentReport.findMany({
where: status ? { status: this.toPrismaReportStatus(status) } : undefined,
include: {
reportedComment: {
include: {
user: {
select: {
id: true,
username: true,
displayName: true,
avatar: true,
},
},
},
},
reporter: {
select: {
id: true,
username: true,
displayName: true,
avatar: true,
},
},
reviewer: {
select: {
id: true,
username: true,
displayName: true,
},
},
},
orderBy: {
createdAt: "desc",
},
});
return reports.map((report) => ({
id: report.id,
reportedCommentId: report.reportedCommentId,
reporterId: report.reporterId,
reason: this.fromPrismaReportReason(report.reason),
details: report.details,
status: this.fromPrismaReportStatus(report.status),
reviewedBy: report.reviewedBy ?? undefined,
reviewNotes: report.reviewNotes ?? undefined,
createdAt: report.createdAt,
updatedAt: report.updatedAt,
reportedComment: {
id: report.reportedComment.id,
content: report.reportedComment.content,
rawContent: report.reportedComment.rawContent ?? undefined,
userId: report.reportedComment.userId,
user: report.reportedComment.user,
},
reporter: report.reporter,
reviewer: report.reviewer ?? undefined,
}));
}
/**
* Get a single comment report by ID (admin only).
*
* @param id - The report ID
* @returns The report or null
*/
async getReportById(id: string): Promise<CommentReportWithDetails | null> {
const report = await this.prisma.commentReport.findUnique({
where: { id },
include: {
reportedComment: {
include: {
user: {
select: {
id: true,
username: true,
displayName: true,
avatar: true,
},
},
},
},
reporter: {
select: {
id: true,
username: true,
displayName: true,
avatar: true,
},
},
reviewer: {
select: {
id: true,
username: true,
displayName: true,
},
},
},
});
if (!report) {
return null;
}
return {
id: report.id,
reportedCommentId: report.reportedCommentId,
reporterId: report.reporterId,
reason: this.fromPrismaReportReason(report.reason),
details: report.details,
status: this.fromPrismaReportStatus(report.status),
reviewedBy: report.reviewedBy ?? undefined,
reviewNotes: report.reviewNotes ?? undefined,
createdAt: report.createdAt,
updatedAt: report.updatedAt,
reportedComment: {
id: report.reportedComment.id,
content: report.reportedComment.content,
rawContent: report.reportedComment.rawContent ?? undefined,
userId: report.reportedComment.userId,
user: report.reportedComment.user,
},
reporter: report.reporter,
reviewer: report.reviewer ?? undefined,
};
}
/**
* Update a comment report's status and review notes (admin only).
*
* @param id - The report ID
* @param reviewerId - The ID of the admin reviewing the report
* @param updateDto - The update details
* @returns The updated report
*/
async updateReport(
id: string,
reviewerId: string,
updateDto: UpdateCommentReportDto,
): Promise<CommentReportWithDetails> {
const report = await this.prisma.commentReport.update({
where: { id },
data: {
status: this.toPrismaReportStatus(updateDto.status),
reviewNotes: updateDto.reviewNotes,
reviewedBy: reviewerId,
},
include: {
reportedComment: {
include: {
user: {
select: {
id: true,
username: true,
displayName: true,
avatar: true,
},
},
},
},
reporter: {
select: {
id: true,
username: true,
displayName: true,
avatar: true,
},
},
reviewer: {
select: {
id: true,
username: true,
displayName: true,
},
},
},
});
return {
id: report.id,
reportedCommentId: report.reportedCommentId,
reporterId: report.reporterId,
reason: this.fromPrismaReportReason(report.reason),
details: report.details,
status: this.fromPrismaReportStatus(report.status),
reviewedBy: report.reviewedBy ?? undefined,
reviewNotes: report.reviewNotes ?? undefined,
createdAt: report.createdAt,
updatedAt: report.updatedAt,
reportedComment: {
id: report.reportedComment.id,
content: report.reportedComment.content,
rawContent: report.reportedComment.rawContent ?? undefined,
userId: report.reportedComment.userId,
user: report.reportedComment.user,
},
reporter: report.reporter,
reviewer: report.reviewer ?? undefined,
};
}
/**
* Check if a comment has any pending reports.
*
* @param commentId - The comment ID
* @returns True if the comment has pending reports
*/
async hasPendingReports(commentId: string): Promise<boolean> {
const count = await this.prisma.commentReport.count({
where: {
reportedCommentId: commentId,
status: PrismaReportStatus.PENDING,
},
});
return count > 0;
}
}
+21 -34
View File
@@ -4,12 +4,11 @@
* @author Naomi Carrigan
*/
import { Comment, CreateCommentDto, PrimaryBadge } from "@library/shared-types";
import { Comment, CreateCommentDto } from "@library/shared-types";
import { prisma } from "../lib/prisma";
import createDOMPurify from "dompurify";
import { JSDOM } from "jsdom";
import { marked } from "marked";
import { validateStringLength, MAX_LENGTHS } from "../utils/validation";
const window = new JSDOM("").window;
const DOMPurify = createDOMPurify(window);
@@ -35,11 +34,6 @@ export class CommentService {
constructor() {}
private sanitizeMarkdown(content: string): string {
// Validate content length before processing
if (!validateStringLength(content, MAX_LENGTHS.COMMENT_CONTENT)) {
throw new Error(`Comment must be ${MAX_LENGTHS.COMMENT_CONTENT} characters or less.`);
}
const html = marked.parse(content, { async: false }) as string;
return DOMPurify.sanitize(html, {
ALLOWED_TAGS: [
@@ -56,12 +50,7 @@ export class CommentService {
});
}
private async mapComment(comment: any): Promise<Comment> {
// Check if comment has pending reports
const hasPendingReports = comment.reports
? comment.reports.some((report: any) => report.status === "PENDING")
: false;
private mapComment(comment: any): Comment {
return {
id: comment.id,
content: comment.content,
@@ -71,7 +60,6 @@ export class CommentService {
id: comment.user.id,
username: comment.user.username,
avatar: comment.user.avatar || undefined,
primaryBadge: (comment.user.primaryBadge as PrimaryBadge) || undefined,
inDiscord: comment.user.inDiscord,
isVip: comment.user.isVip,
isMod: comment.user.isMod,
@@ -83,7 +71,6 @@ export class CommentService {
artId: comment.artId || undefined,
showId: comment.showId || undefined,
mangaId: comment.mangaId || undefined,
hasPendingReports,
createdAt: comment.createdAt,
updatedAt: comment.updatedAt,
};
@@ -92,28 +79,28 @@ export class CommentService {
async getCommentsForGame(gameId: string): Promise<Comment[]> {
const comments = await this.prisma.comment.findMany({
where: { gameId },
include: { user: true, reports: true },
include: { user: true },
orderBy: { createdAt: "desc" },
});
return Promise.all(comments.map((c) => this.mapComment(c)));
return comments.map((c) => this.mapComment(c));
}
async getCommentsForBook(bookId: string): Promise<Comment[]> {
const comments = await this.prisma.comment.findMany({
where: { bookId },
include: { user: true, reports: true },
include: { user: true },
orderBy: { createdAt: "desc" },
});
return Promise.all(comments.map((c) => this.mapComment(c)));
return comments.map((c) => this.mapComment(c));
}
async getCommentsForMusic(musicId: string): Promise<Comment[]> {
const comments = await this.prisma.comment.findMany({
where: { musicId },
include: { user: true, reports: true },
include: { user: true },
orderBy: { createdAt: "desc" },
});
return Promise.all(comments.map((c) => this.mapComment(c)));
return comments.map((c) => this.mapComment(c));
}
async createCommentForGame(
@@ -129,7 +116,7 @@ export class CommentService {
userId,
gameId,
},
include: { user: true, reports: true },
include: { user: true },
});
return this.mapComment(comment);
}
@@ -147,7 +134,7 @@ export class CommentService {
userId,
bookId,
},
include: { user: true, reports: true },
include: { user: true },
});
return this.mapComment(comment);
}
@@ -165,7 +152,7 @@ export class CommentService {
userId,
musicId,
},
include: { user: true, reports: true },
include: { user: true },
});
return this.mapComment(comment);
}
@@ -173,10 +160,10 @@ export class CommentService {
async getCommentsForArt(artId: string): Promise<Comment[]> {
const comments = await this.prisma.comment.findMany({
where: { artId },
include: { user: true, reports: true },
include: { user: true },
orderBy: { createdAt: "desc" },
});
return Promise.all(comments.map((c) => this.mapComment(c)));
return comments.map((c) => this.mapComment(c));
}
async createCommentForArt(
@@ -192,7 +179,7 @@ export class CommentService {
userId,
artId,
},
include: { user: true, reports: true },
include: { user: true },
});
return this.mapComment(comment);
}
@@ -200,10 +187,10 @@ export class CommentService {
async getCommentsForShow(showId: string): Promise<Comment[]> {
const comments = await this.prisma.comment.findMany({
where: { showId },
include: { user: true, reports: true },
include: { user: true },
orderBy: { createdAt: "desc" },
});
return Promise.all(comments.map((c) => this.mapComment(c)));
return comments.map((c) => this.mapComment(c));
}
async createCommentForShow(
@@ -219,7 +206,7 @@ export class CommentService {
userId,
showId,
},
include: { user: true, reports: true },
include: { user: true },
});
return this.mapComment(comment);
}
@@ -227,10 +214,10 @@ export class CommentService {
async getCommentsForManga(mangaId: string): Promise<Comment[]> {
const comments = await this.prisma.comment.findMany({
where: { mangaId },
include: { user: true, reports: true },
include: { user: true },
orderBy: { createdAt: "desc" },
});
return Promise.all(comments.map((c) => this.mapComment(c)));
return comments.map((c) => this.mapComment(c));
}
async createCommentForManga(
@@ -246,7 +233,7 @@ export class CommentService {
userId,
mangaId,
},
include: { user: true, reports: true },
include: { user: true },
});
return this.mapComment(comment);
}
@@ -269,7 +256,7 @@ export class CommentService {
content: sanitizedContent,
rawContent: content,
},
include: { user: true, reports: true },
include: { user: true },
});
return this.mapComment(comment);
}
-115
View File
@@ -6,90 +6,12 @@
import { Game, GameStatus, CreateGameDto, UpdateGameDto } from "@library/shared-types";
import { prisma } from "../lib/prisma";
import {
validateUrl,
validateRating,
validateStringLength,
validateDataUrl,
MAX_LENGTHS,
} from "../utils/validation";
export class GameService {
private prisma = prisma;
constructor() {}
/**
* Validate game data for security.
*/
private validateGameData(data: CreateGameDto | UpdateGameDto): void {
// Validate string lengths
if (!validateStringLength(data.title, MAX_LENGTHS.TITLE)) {
throw new Error(`Title must be ${MAX_LENGTHS.TITLE} characters or less.`);
}
if (!validateStringLength(data.platform, MAX_LENGTHS.AUTHOR)) {
throw new Error(`Platform must be ${MAX_LENGTHS.AUTHOR} characters or less.`);
}
if (!validateStringLength(data.notes, MAX_LENGTHS.NOTES)) {
throw new Error(`Notes must be ${MAX_LENGTHS.NOTES} characters or less.`);
}
// Validate rating
if (!validateRating(data.rating)) {
throw new Error("Rating must be an integer between 0 and 10.");
}
// Validate cover image URL
if (data.coverImage) {
if (data.coverImage.startsWith("data:")) {
// Extract just the base64 data (after the comma)
const base64Data = data.coverImage.split(",")[1];
if (!base64Data) {
throw new Error("Invalid image data URL format.");
}
// Calculate decoded size: base64 is ~4/3 larger than original, so multiply by 0.75
const decodedSizeInBytes = base64Data.length * 0.75;
if (decodedSizeInBytes > MAX_LENGTHS.DATA_URL) {
throw new Error("Cover image must be under 5MB.");
}
if (!validateDataUrl(data.coverImage)) {
throw new Error("Invalid image data URL.");
}
} else {
if (!validateStringLength(data.coverImage, MAX_LENGTHS.URL)) {
throw new Error(`Cover image URL must be ${MAX_LENGTHS.URL} characters or less.`);
}
if (!validateUrl(data.coverImage)) {
throw new Error("Invalid cover image URL. Only http and https URLs are allowed.");
}
}
}
// Validate tags
if (data.tags) {
for (const tag of data.tags) {
if (!validateStringLength(tag, MAX_LENGTHS.TAGS)) {
throw new Error(`Each tag must be ${MAX_LENGTHS.TAGS} characters or less.`);
}
}
}
// Validate link URLs
if (data.links) {
for (const link of data.links) {
if (!validateUrl(link.url)) {
throw new Error(`Invalid link URL: ${link.title}. Only http and https URLs are allowed.`);
}
if (!validateStringLength(link.title, MAX_LENGTHS.TITLE)) {
throw new Error(`Link title must be ${MAX_LENGTHS.TITLE} characters or less.`);
}
if (!validateStringLength(link.url, MAX_LENGTHS.URL)) {
throw new Error(`Link URL must be ${MAX_LENGTHS.URL} characters or less.`);
}
}
}
}
/**
* Get all games.
*/
@@ -102,9 +24,7 @@ export class GameService {
...game,
status: game.status as unknown as GameStatus,
dateAdded: game.dateAdded,
dateStarted: game.dateStarted || undefined,
dateCompleted: game.dateCompleted || undefined,
dateFinished: game.dateFinished || undefined,
tags: game.tags ?? [],
links: game.links ?? [],
createdAt: game.createdAt,
@@ -126,9 +46,7 @@ export class GameService {
...game,
status: game.status as unknown as GameStatus,
dateAdded: game.dateAdded,
dateStarted: game.dateStarted || undefined,
dateCompleted: game.dateCompleted || undefined,
dateFinished: game.dateFinished || undefined,
tags: game.tags ?? [],
links: game.links ?? [],
createdAt: game.createdAt,
@@ -136,36 +54,10 @@ export class GameService {
};
}
/**
* Get all games in a series, ordered by seriesOrder.
*/
async getGamesBySeries(seriesName: string): Promise<Game[]> {
const games = await this.prisma.game.findMany({
where: { series: seriesName },
orderBy: { seriesOrder: "asc" },
});
return games.map((game) => ({
...game,
status: game.status as unknown as GameStatus,
dateAdded: game.dateAdded,
dateStarted: game.dateStarted || undefined,
dateCompleted: game.dateCompleted || undefined,
dateFinished: game.dateFinished || undefined,
tags: game.tags ?? [],
links: game.links ?? [],
createdAt: game.createdAt,
updatedAt: game.updatedAt,
}));
}
/**
* Create new game.
*/
async createGame(data: CreateGameDto): Promise<Game> {
// Validate input
this.validateGameData(data);
const game = await this.prisma.game.create({
data: {
...data,
@@ -177,9 +69,7 @@ export class GameService {
...game,
status: game.status as unknown as GameStatus,
dateAdded: game.dateAdded,
dateStarted: game.dateStarted || undefined,
dateCompleted: game.dateCompleted || undefined,
dateFinished: game.dateFinished || undefined,
tags: game.tags ?? [],
links: game.links ?? [],
createdAt: game.createdAt,
@@ -191,9 +81,6 @@ export class GameService {
* Update game by ID.
*/
async updateGame(id: string, data: UpdateGameDto): Promise<Game> {
// Validate input
this.validateGameData(data);
const updateData = { ...data };
if (updateData.status) {
updateData.status = updateData.status.toUpperCase() as any;
@@ -208,9 +95,7 @@ export class GameService {
...game,
status: game.status as unknown as GameStatus,
dateAdded: game.dateAdded,
dateStarted: game.dateStarted || undefined,
dateCompleted: game.dateCompleted || undefined,
dateFinished: game.dateFinished || undefined,
tags: game.tags ?? [],
links: game.links ?? [],
createdAt: game.createdAt,
-222
View File
@@ -1,222 +0,0 @@
/**
* @copyright 2026 NHCarrigan
* @license Naomi's Public License
* @author Naomi Carrigan
*/
import type {
LeaderboardResponse,
SuggestionsLeaderboard,
LikesLeaderboard,
CommentsLeaderboard,
OverallLeaderboard,
} from "@library/shared-types";
import { prisma } from "../lib/prisma";
export class LeaderboardService {
private prisma = prisma;
constructor() {}
/**
* Get top users by suggestions submitted and accepted.
*/
async getTopSuggestions(limit = 25): Promise<SuggestionsLeaderboard[]> {
const users = await this.prisma.user.findMany({
where: { profilePublic: true, isBanned: false },
include: {
suggestions: {
select: {
status: true,
},
},
},
});
const leaderboard = users
.map((user) => {
const totalSuggestions = user.suggestions.length;
const acceptedSuggestions = user.suggestions.filter(
(s) => s.status === "ACCEPTED"
).length;
const acceptanceRate =
totalSuggestions > 0
? Math.round((acceptedSuggestions / totalSuggestions) * 100)
: 0;
return {
id: user.id,
username: user.username,
slug: user.slug,
avatar: user.avatar,
primaryBadge: user.primaryBadge,
isVip: user.isVip,
isMod: user.isMod,
isStaff: user.isStaff,
createdAt: user.createdAt,
totalSuggestions,
acceptedSuggestions,
acceptanceRate,
};
})
.filter((user) => user.totalSuggestions > 0)
.sort((a, b) => {
if (b.totalSuggestions !== a.totalSuggestions) {
return b.totalSuggestions - a.totalSuggestions;
}
return b.acceptanceRate - a.acceptanceRate;
})
.slice(0, limit);
return leaderboard;
}
/**
* Get top users by likes given.
*/
async getTopLikes(limit = 25): Promise<LikesLeaderboard[]> {
const users = await this.prisma.user.findMany({
where: { profilePublic: true, isBanned: false },
include: {
likes: true,
},
});
const leaderboard = users
.map((user) => ({
id: user.id,
username: user.username,
slug: user.slug,
avatar: user.avatar,
primaryBadge: user.primaryBadge,
isVip: user.isVip,
isMod: user.isMod,
isStaff: user.isStaff,
createdAt: user.createdAt,
totalLikes: user.likes.length,
}))
.filter((user) => user.totalLikes > 0)
.sort((a, b) => b.totalLikes - a.totalLikes)
.slice(0, limit);
return leaderboard;
}
/**
* Get top users by comments posted.
*/
async getTopComments(limit = 25): Promise<CommentsLeaderboard[]> {
const users = await this.prisma.user.findMany({
where: { profilePublic: true, isBanned: false },
include: {
comments: true,
},
});
const leaderboard = users
.map((user) => ({
id: user.id,
username: user.username,
slug: user.slug,
avatar: user.avatar,
primaryBadge: user.primaryBadge,
isVip: user.isVip,
isMod: user.isMod,
isStaff: user.isStaff,
createdAt: user.createdAt,
totalComments: user.comments.length,
}))
.filter((user) => user.totalComments > 0)
.sort((a, b) => b.totalComments - a.totalComments)
.slice(0, limit);
return leaderboard;
}
/**
* Get overall leaderboard based on combined engagement.
*/
async getOverallLeaderboard(limit = 25): Promise<OverallLeaderboard[]> {
const users = await this.prisma.user.findMany({
where: { profilePublic: true, isBanned: false },
include: {
suggestions: true,
likes: true,
comments: true,
userAchievements: true,
},
});
const leaderboard = users
.map((user) => {
const totalSuggestions = user.suggestions.length;
const totalLikes = user.likes.length;
const totalComments = user.comments.length;
const achievementCount = user.userAchievements.length;
const diversityScore =
(totalSuggestions > 0 ? 1 : 0) +
(totalLikes > 0 ? 1 : 0) +
(totalComments > 0 ? 1 : 0);
return {
id: user.id,
username: user.username,
slug: user.slug,
avatar: user.avatar,
primaryBadge: user.primaryBadge,
isVip: user.isVip,
isMod: user.isMod,
isStaff: user.isStaff,
createdAt: user.createdAt,
totalSuggestions,
totalLikes,
totalComments,
achievementCount,
achievementPoints: user.achievementPoints,
currentStreak: user.currentStreak,
diversityScore,
};
})
.filter(
(user) =>
user.totalSuggestions > 0 ||
user.totalLikes > 0 ||
user.totalComments > 0
)
.sort((a, b) => {
if (b.achievementPoints !== a.achievementPoints) {
return b.achievementPoints - a.achievementPoints;
}
if (b.diversityScore !== a.diversityScore) {
return b.diversityScore - a.diversityScore;
}
const totalA = a.totalSuggestions + a.totalLikes + a.totalComments;
const totalB = b.totalSuggestions + b.totalLikes + b.totalComments;
return totalB - totalA;
})
.slice(0, limit);
return leaderboard;
}
/**
* Get all leaderboards at once.
*/
async getAllLeaderboards(limit = 25): Promise<LeaderboardResponse> {
const [topSuggestions, topLikes, topComments, topOverall] =
await Promise.all([
this.getTopSuggestions(limit),
this.getTopLikes(limit),
this.getTopComments(limit),
this.getOverallLeaderboard(limit),
]);
return {
topSuggestions,
topLikes,
topComments,
topOverall,
};
}
}
+5 -14
View File
@@ -7,9 +7,8 @@
import type { FastifyRequest } from 'fastify';
import { prisma } from '../lib/prisma';
import { AuditService } from './audit.service';
import { AchievementService } from './achievement.service';
import type { Like, LikeCountDto, LikedItemDto, LikeResponse } from '@library/shared-types';
import { AuditAction, AuditCategory, AchievementCategory } from '@library/shared-types';
import { AuditAction, AuditCategory } from '@library/shared-types';
export class LikeService {
async toggleLike(userId: string, entityType: Like['entityType'], entityId: string, req: FastifyRequest): Promise<LikeResponse> {
@@ -33,8 +32,8 @@ export class LikeService {
});
await AuditService.logFromRequest(req, {
action: AuditAction.unlike,
category: AuditCategory.content,
action: AuditAction.UNLIKE,
category: AuditCategory.CONTENT,
resourceType: entityType,
resourceId: entityId,
details: `Unliked ${entityType}`
@@ -53,21 +52,13 @@ export class LikeService {
});
await AuditService.logFromRequest(req, {
action: AuditAction.like,
category: AuditCategory.content,
action: AuditAction.LIKE,
category: AuditCategory.CONTENT,
resourceType: entityType,
resourceId: entityId,
details: `Liked ${entityType}`
});
// Check for like achievements
const achievementService = new AchievementService();
await achievementService.checkAchievements(
userId,
AchievementCategory.Like,
req
);
const count = await this.getLikeCount(entityType, entityId);
return { liked: true, count };
}
-89
View File
@@ -6,87 +6,12 @@
import { Manga, MangaStatus, CreateMangaDto, UpdateMangaDto } from "@library/shared-types";
import { prisma } from "../lib/prisma";
import {
validateUrl,
validateRating,
validateStringLength,
validateDataUrl,
MAX_LENGTHS,
} from "../utils/validation";
export class MangaService {
private prisma = prisma;
constructor() {}
/**
* Validate manga data for security.
*/
private validateMangaData(data: CreateMangaDto | UpdateMangaDto): void {
// Validate string lengths
if (!validateStringLength(data.title, MAX_LENGTHS.TITLE)) {
throw new Error(`Title must be ${MAX_LENGTHS.TITLE} characters or less.`);
}
if (!validateStringLength(data.author, MAX_LENGTHS.AUTHOR)) {
throw new Error(`Author must be ${MAX_LENGTHS.AUTHOR} characters or less.`);
}
if (!validateStringLength(data.notes, MAX_LENGTHS.NOTES)) {
throw new Error(`Notes must be ${MAX_LENGTHS.NOTES} characters or less.`);
}
// Validate rating
if (!validateRating(data.rating)) {
throw new Error("Rating must be an integer between 0 and 10.");
}
// Validate cover image URL
if (data.coverImage) {
if (data.coverImage.startsWith("data:")) {
const base64Data = data.coverImage.split(",")[1];
if (!base64Data) {
throw new Error("Invalid image data URL format.");
}
const sizeInBytes = base64Data.length * 0.75;
if (sizeInBytes > MAX_LENGTHS.DATA_URL) {
throw new Error("Cover image must be under 5MB.");
}
if (!validateDataUrl(data.coverImage)) {
throw new Error("Invalid image data URL.");
}
} else {
if (!validateStringLength(data.coverImage, MAX_LENGTHS.URL)) {
throw new Error(`Cover image URL must be ${MAX_LENGTHS.URL} characters or less.`);
}
if (!validateUrl(data.coverImage)) {
throw new Error("Invalid cover image URL. Only http and https URLs are allowed.");
}
}
}
// Validate tags
if (data.tags) {
for (const tag of data.tags) {
if (!validateStringLength(tag, MAX_LENGTHS.TAGS)) {
throw new Error(`Each tag must be ${MAX_LENGTHS.TAGS} characters or less.`);
}
}
}
// Validate link URLs
if (data.links) {
for (const link of data.links) {
if (!validateUrl(link.url)) {
throw new Error(`Invalid link URL: ${link.title}. Only http and https URLs are allowed.`);
}
if (!validateStringLength(link.title, MAX_LENGTHS.TITLE)) {
throw new Error(`Link title must be ${MAX_LENGTHS.TITLE} characters or less.`);
}
if (!validateStringLength(link.url, MAX_LENGTHS.URL)) {
throw new Error(`Link URL must be ${MAX_LENGTHS.URL} characters or less.`);
}
}
}
}
async getAllManga(): Promise<Manga[]> {
const manga = await this.prisma.manga.findMany({
orderBy: { updatedAt: "desc" },
@@ -96,9 +21,7 @@ export class MangaService {
...m,
status: m.status as unknown as MangaStatus,
dateAdded: m.dateAdded,
dateStarted: m.dateStarted || undefined,
dateCompleted: m.dateCompleted || undefined,
dateFinished: m.dateFinished || undefined,
tags: m.tags ?? [],
links: m.links ?? [],
createdAt: m.createdAt,
@@ -117,9 +40,7 @@ export class MangaService {
...manga,
status: manga.status as unknown as MangaStatus,
dateAdded: manga.dateAdded,
dateStarted: manga.dateStarted || undefined,
dateCompleted: manga.dateCompleted || undefined,
dateFinished: manga.dateFinished || undefined,
tags: manga.tags ?? [],
links: manga.links ?? [],
createdAt: manga.createdAt,
@@ -128,9 +49,6 @@ export class MangaService {
}
async createManga(data: CreateMangaDto): Promise<Manga> {
// Validate input
this.validateMangaData(data);
const manga = await this.prisma.manga.create({
data: {
...data,
@@ -142,9 +60,7 @@ export class MangaService {
...manga,
status: manga.status as unknown as MangaStatus,
dateAdded: manga.dateAdded,
dateStarted: manga.dateStarted || undefined,
dateCompleted: manga.dateCompleted || undefined,
dateFinished: manga.dateFinished || undefined,
tags: manga.tags ?? [],
links: manga.links ?? [],
createdAt: manga.createdAt,
@@ -153,9 +69,6 @@ export class MangaService {
}
async updateManga(id: string, data: UpdateMangaDto): Promise<Manga> {
// Validate input
this.validateMangaData(data);
const updateData = { ...data };
if (updateData.status) {
updateData.status = updateData.status.toUpperCase() as any;
@@ -170,9 +83,7 @@ export class MangaService {
...manga,
status: manga.status as unknown as MangaStatus,
dateAdded: manga.dateAdded,
dateStarted: manga.dateStarted || undefined,
dateCompleted: manga.dateCompleted || undefined,
dateFinished: manga.dateFinished || undefined,
tags: manga.tags ?? [],
links: manga.links ?? [],
createdAt: manga.createdAt,
-89
View File
@@ -6,87 +6,12 @@
import { Music, MusicStatus, MusicType, CreateMusicDto, UpdateMusicDto } from "@library/shared-types";
import { prisma } from "../lib/prisma";
import {
validateUrl,
validateRating,
validateStringLength,
validateDataUrl,
MAX_LENGTHS,
} from "../utils/validation";
export class MusicService {
private prisma = prisma;
constructor() {}
/**
* Validate music data for security.
*/
private validateMusicData(data: CreateMusicDto | UpdateMusicDto): void {
// Validate string lengths
if (!validateStringLength(data.title, MAX_LENGTHS.TITLE)) {
throw new Error(`Title must be ${MAX_LENGTHS.TITLE} characters or less.`);
}
if (!validateStringLength(data.artist, MAX_LENGTHS.AUTHOR)) {
throw new Error(`Artist must be ${MAX_LENGTHS.AUTHOR} characters or less.`);
}
if (!validateStringLength(data.notes, MAX_LENGTHS.NOTES)) {
throw new Error(`Notes must be ${MAX_LENGTHS.NOTES} characters or less.`);
}
// Validate rating
if (data.rating !== undefined && !validateRating(data.rating)) {
throw new Error("Rating must be an integer between 0 and 10.");
}
// Validate cover art URL
if (data.coverArt) {
if (data.coverArt.startsWith("data:")) {
const base64Data = data.coverArt.split(",")[1];
if (!base64Data) {
throw new Error("Invalid image data URL format.");
}
const sizeInBytes = base64Data.length * 0.75;
if (sizeInBytes > MAX_LENGTHS.DATA_URL) {
throw new Error("Cover image must be under 5MB.");
}
if (!validateDataUrl(data.coverArt)) {
throw new Error("Invalid image data URL.");
}
} else {
if (!validateStringLength(data.coverArt, MAX_LENGTHS.URL)) {
throw new Error(`Cover image URL must be ${MAX_LENGTHS.URL} characters or less.`);
}
if (!validateUrl(data.coverArt)) {
throw new Error("Invalid cover image URL. Only http and https URLs are allowed.");
}
}
}
// Validate tags
if (data.tags) {
for (const tag of data.tags) {
if (!validateStringLength(tag, MAX_LENGTHS.TAGS)) {
throw new Error(`Each tag must be ${MAX_LENGTHS.TAGS} characters or less.`);
}
}
}
// Validate link URLs
if (data.links) {
for (const link of data.links) {
if (!validateUrl(link.url)) {
throw new Error(`Invalid link URL: ${link.title}. Only http and https URLs are allowed.`);
}
if (!validateStringLength(link.title, MAX_LENGTHS.TITLE)) {
throw new Error(`Link title must be ${MAX_LENGTHS.TITLE} characters or less.`);
}
if (!validateStringLength(link.url, MAX_LENGTHS.URL)) {
throw new Error(`Link URL must be ${MAX_LENGTHS.URL} characters or less.`);
}
}
}
}
/**
* Get all music.
*/
@@ -100,9 +25,7 @@ export class MusicService {
type: music.type as unknown as MusicType,
status: music.status as unknown as MusicStatus,
dateAdded: music.dateAdded,
dateStarted: music.dateStarted || undefined,
dateCompleted: music.dateCompleted || undefined,
dateFinished: music.dateFinished || undefined,
tags: music.tags ?? [],
links: music.links ?? [],
createdAt: music.createdAt,
@@ -125,9 +48,7 @@ export class MusicService {
type: music.type as unknown as MusicType,
status: music.status as unknown as MusicStatus,
dateAdded: music.dateAdded,
dateStarted: music.dateStarted || undefined,
dateCompleted: music.dateCompleted || undefined,
dateFinished: music.dateFinished || undefined,
tags: music.tags ?? [],
links: music.links ?? [],
createdAt: music.createdAt,
@@ -139,9 +60,6 @@ export class MusicService {
* Create new music.
*/
async createMusic(data: CreateMusicDto): Promise<Music> {
// Validate input
this.validateMusicData(data);
const music = await this.prisma.music.create({
data: {
...data,
@@ -155,9 +73,7 @@ export class MusicService {
type: music.type as unknown as MusicType,
status: music.status as unknown as MusicStatus,
dateAdded: music.dateAdded,
dateStarted: music.dateStarted || undefined,
dateCompleted: music.dateCompleted || undefined,
dateFinished: music.dateFinished || undefined,
tags: music.tags ?? [],
links: music.links ?? [],
createdAt: music.createdAt,
@@ -169,9 +85,6 @@ export class MusicService {
* Update music by ID.
*/
async updateMusic(id: string, data: UpdateMusicDto): Promise<Music> {
// Validate input
this.validateMusicData(data);
const updateData = { ...data };
if (updateData.type) {
updateData.type = updateData.type.toUpperCase() as any;
@@ -190,9 +103,7 @@ export class MusicService {
type: music.type as unknown as MusicType,
status: music.status as unknown as MusicStatus,
dateAdded: music.dateAdded,
dateStarted: music.dateStarted || undefined,
dateCompleted: music.dateCompleted || undefined,
dateFinished: music.dateFinished || undefined,
tags: music.tags ?? [],
links: music.links ?? [],
createdAt: music.createdAt,
-312
View File
@@ -1,312 +0,0 @@
/**
* @copyright NHCarrigan
* @license Naomi's Public License
* @author Naomi Carrigan
*/
import {
ReportStatus as PrismaReportStatus,
ReportReason as PrismaReportReason,
} from "@prisma/client";
import type {
CreateReportDto,
ProfileReportWithUsers,
ReportStatus,
UpdateReportDto,
} from "@library/shared-types";
import { ReportReason } from "@library/shared-types";
import { prisma } from "../lib/prisma.js";
export class ReportService {
private prisma = prisma;
/**
* Convert Prisma ReportReason to shared-types ReportReason
*/
private toPrismaReportReason(reason: ReportReason): PrismaReportReason {
return reason as unknown as PrismaReportReason;
}
/**
* Convert Prisma ReportStatus to shared-types ReportStatus
*/
private toPrismaReportStatus(status: ReportStatus): PrismaReportStatus {
return status as unknown as PrismaReportStatus;
}
/**
* Convert Prisma enum back to shared-types enum
*/
private fromPrismaReportReason(reason: PrismaReportReason): ReportReason {
return reason as unknown as ReportReason;
}
/**
* Convert Prisma enum back to shared-types enum
*/
private fromPrismaReportStatus(status: PrismaReportStatus): ReportStatus {
return status as unknown as ReportStatus;
}
/**
* Create a new profile report.
*
* @param reporterId - The ID of the user making the report
* @param createDto - The report details
* @returns The created report
* @throws Error if user already has a pending report for this profile
*/
async createReport(
reporterId: string,
createDto: CreateReportDto,
): Promise<ProfileReportWithUsers> {
// Check if user already has a pending report for this profile
const existingReport = await this.prisma.profileReport.findFirst({
where: {
reporterId,
reportedUserId: createDto.reportedUserId,
status: PrismaReportStatus.PENDING,
},
});
if (existingReport) {
throw new Error(
"You already have a pending report for this profile. Please wait for it to be reviewed.",
);
}
// Check if user has reached the limit of pending reports (5 max)
const pendingReportsCount = await this.prisma.profileReport.count({
where: {
reporterId,
status: PrismaReportStatus.PENDING,
},
});
if (pendingReportsCount >= 5) {
throw new Error(
"You have reached the maximum number of pending reports (5). Please wait for your existing reports to be reviewed.",
);
}
const report = await this.prisma.profileReport.create({
data: {
reporterId,
reportedUserId: createDto.reportedUserId,
reason: this.toPrismaReportReason(createDto.reason),
details: createDto.details,
},
include: {
reportedUser: {
select: {
id: true,
username: true,
displayName: true,
avatar: true,
},
},
reporter: {
select: {
id: true,
username: true,
displayName: true,
avatar: true,
},
},
},
});
return {
id: report.id,
reportedUserId: report.reportedUserId,
reporterId: report.reporterId,
reason: this.fromPrismaReportReason(report.reason),
details: report.details,
status: this.fromPrismaReportStatus(report.status),
reviewedBy: report.reviewedBy ?? undefined,
reviewNotes: report.reviewNotes ?? undefined,
createdAt: report.createdAt,
updatedAt: report.updatedAt,
reportedUser: report.reportedUser,
reporter: report.reporter,
};
}
/**
* Get all reports (admin only). Optionally filter by status.
*
* @param status - Optional status filter
* @returns All reports matching the filter
*/
async getAllReports(
status?: ReportStatus,
): Promise<ProfileReportWithUsers[]> {
const reports = await this.prisma.profileReport.findMany({
where: status ? { status: this.toPrismaReportStatus(status) } : undefined,
include: {
reportedUser: {
select: {
id: true,
username: true,
displayName: true,
avatar: true,
},
},
reporter: {
select: {
id: true,
username: true,
displayName: true,
avatar: true,
},
},
reviewer: {
select: {
id: true,
username: true,
displayName: true,
},
},
},
orderBy: {
createdAt: "desc",
},
});
return reports.map((report) => ({
id: report.id,
reportedUserId: report.reportedUserId,
reporterId: report.reporterId,
reason: this.fromPrismaReportReason(report.reason),
details: report.details,
status: this.fromPrismaReportStatus(report.status),
reviewedBy: report.reviewedBy ?? undefined,
reviewNotes: report.reviewNotes ?? undefined,
createdAt: report.createdAt,
updatedAt: report.updatedAt,
reportedUser: report.reportedUser,
reporter: report.reporter,
reviewer: report.reviewer ?? undefined,
}));
}
/**
* Get a single report by ID (admin only).
*
* @param id - The report ID
* @returns The report or null
*/
async getReportById(id: string): Promise<ProfileReportWithUsers | null> {
const report = await this.prisma.profileReport.findUnique({
where: { id },
include: {
reportedUser: {
select: {
id: true,
username: true,
displayName: true,
avatar: true,
},
},
reporter: {
select: {
id: true,
username: true,
displayName: true,
avatar: true,
},
},
reviewer: {
select: {
id: true,
username: true,
displayName: true,
},
},
},
});
if (!report) {
return null;
}
return {
id: report.id,
reportedUserId: report.reportedUserId,
reporterId: report.reporterId,
reason: this.fromPrismaReportReason(report.reason),
details: report.details,
status: this.fromPrismaReportStatus(report.status),
reviewedBy: report.reviewedBy ?? undefined,
reviewNotes: report.reviewNotes ?? undefined,
createdAt: report.createdAt,
updatedAt: report.updatedAt,
reportedUser: report.reportedUser,
reporter: report.reporter,
reviewer: report.reviewer ?? undefined,
};
}
/**
* Update a report's status and review notes (admin only).
*
* @param id - The report ID
* @param reviewerId - The ID of the admin reviewing the report
* @param updateDto - The update details
* @returns The updated report
*/
async updateReport(
id: string,
reviewerId: string,
updateDto: UpdateReportDto,
): Promise<ProfileReportWithUsers> {
const report = await this.prisma.profileReport.update({
where: { id },
data: {
status: this.toPrismaReportStatus(updateDto.status),
reviewNotes: updateDto.reviewNotes,
reviewedBy: reviewerId,
},
include: {
reportedUser: {
select: {
id: true,
username: true,
displayName: true,
avatar: true,
},
},
reporter: {
select: {
id: true,
username: true,
displayName: true,
avatar: true,
},
},
reviewer: {
select: {
id: true,
username: true,
displayName: true,
},
},
},
});
return {
id: report.id,
reportedUserId: report.reportedUserId,
reporterId: report.reporterId,
reason: this.fromPrismaReportReason(report.reason),
details: report.details,
status: this.fromPrismaReportStatus(report.status),
reviewedBy: report.reviewedBy ?? undefined,
reviewNotes: report.reviewNotes ?? undefined,
createdAt: report.createdAt,
updatedAt: report.updatedAt,
reportedUser: report.reportedUser,
reporter: report.reporter,
reviewer: report.reviewer ?? undefined,
};
}
}
-86
View File
@@ -6,84 +6,12 @@
import { Show, ShowStatus, ShowType, CreateShowDto, UpdateShowDto } from "@library/shared-types";
import { prisma } from "../lib/prisma";
import {
validateUrl,
validateRating,
validateStringLength,
validateDataUrl,
MAX_LENGTHS,
} from "../utils/validation";
export class ShowService {
private prisma = prisma;
constructor() {}
/**
* Validate show data for security.
*/
private validateShowData(data: CreateShowDto | UpdateShowDto): void {
// Validate string lengths
if (!validateStringLength(data.title, MAX_LENGTHS.TITLE)) {
throw new Error(`Title must be ${MAX_LENGTHS.TITLE} characters or less.`);
}
if (!validateStringLength(data.notes, MAX_LENGTHS.NOTES)) {
throw new Error(`Notes must be ${MAX_LENGTHS.NOTES} characters or less.`);
}
// Validate rating
if (!validateRating(data.rating)) {
throw new Error("Rating must be an integer between 0 and 10.");
}
// Validate cover image URL
if (data.coverImage) {
if (data.coverImage.startsWith("data:")) {
const base64Data = data.coverImage.split(",")[1];
if (!base64Data) {
throw new Error("Invalid image data URL format.");
}
const sizeInBytes = base64Data.length * 0.75;
if (sizeInBytes > MAX_LENGTHS.DATA_URL) {
throw new Error("Cover image must be under 5MB.");
}
if (!validateDataUrl(data.coverImage)) {
throw new Error("Invalid image data URL.");
}
} else {
if (!validateStringLength(data.coverImage, MAX_LENGTHS.URL)) {
throw new Error(`Cover image URL must be ${MAX_LENGTHS.URL} characters or less.`);
}
if (!validateUrl(data.coverImage)) {
throw new Error("Invalid cover image URL. Only http and https URLs are allowed.");
}
}
}
// Validate tags
if (data.tags) {
for (const tag of data.tags) {
if (!validateStringLength(tag, MAX_LENGTHS.TAGS)) {
throw new Error(`Each tag must be ${MAX_LENGTHS.TAGS} characters or less.`);
}
}
}
// Validate link URLs
if (data.links) {
for (const link of data.links) {
if (!validateUrl(link.url)) {
throw new Error(`Invalid link URL: ${link.title}. Only http and https URLs are allowed.`);
}
if (!validateStringLength(link.title, MAX_LENGTHS.TITLE)) {
throw new Error(`Link title must be ${MAX_LENGTHS.TITLE} characters or less.`);
}
if (!validateStringLength(link.url, MAX_LENGTHS.URL)) {
throw new Error(`Link URL must be ${MAX_LENGTHS.URL} characters or less.`);
}
}
}
}
async getAllShows(): Promise<Show[]> {
const shows = await this.prisma.show.findMany({
orderBy: { updatedAt: "desc" },
@@ -94,9 +22,7 @@ export class ShowService {
type: show.type as unknown as ShowType,
status: show.status as unknown as ShowStatus,
dateAdded: show.dateAdded,
dateStarted: show.dateStarted || undefined,
dateCompleted: show.dateCompleted || undefined,
dateFinished: show.dateFinished || undefined,
tags: show.tags ?? [],
links: show.links ?? [],
createdAt: show.createdAt,
@@ -116,9 +42,7 @@ export class ShowService {
type: show.type as unknown as ShowType,
status: show.status as unknown as ShowStatus,
dateAdded: show.dateAdded,
dateStarted: show.dateStarted || undefined,
dateCompleted: show.dateCompleted || undefined,
dateFinished: show.dateFinished || undefined,
tags: show.tags ?? [],
links: show.links ?? [],
createdAt: show.createdAt,
@@ -127,9 +51,6 @@ export class ShowService {
}
async createShow(data: CreateShowDto): Promise<Show> {
// Validate input
this.validateShowData(data);
const show = await this.prisma.show.create({
data: {
...data,
@@ -143,9 +64,7 @@ export class ShowService {
type: show.type as unknown as ShowType,
status: show.status as unknown as ShowStatus,
dateAdded: show.dateAdded,
dateStarted: show.dateStarted || undefined,
dateCompleted: show.dateCompleted || undefined,
dateFinished: show.dateFinished || undefined,
tags: show.tags ?? [],
links: show.links ?? [],
createdAt: show.createdAt,
@@ -154,9 +73,6 @@ export class ShowService {
}
async updateShow(id: string, data: UpdateShowDto): Promise<Show> {
// Validate input
this.validateShowData(data);
const updateData = { ...data };
if (updateData.type) {
updateData.type = updateData.type.toUpperCase() as any;
@@ -175,9 +91,7 @@ export class ShowService {
type: show.type as unknown as ShowType,
status: show.status as unknown as ShowStatus,
dateAdded: show.dateAdded,
dateStarted: show.dateStarted || undefined,
dateCompleted: show.dateCompleted || undefined,
dateFinished: show.dateFinished || undefined,
tags: show.tags ?? [],
links: show.links ?? [],
createdAt: show.createdAt,
+1 -264
View File
@@ -4,15 +4,8 @@
* @author Naomi Carrigan
*/
import { User, PrimaryBadge } from "@library/shared-types";
import { User } from "@library/shared-types";
import { prisma } from "../lib/prisma";
import { SuggestionStatus } from "@prisma/client";
import {
validateUrl,
validateSlug,
validateStringLength,
MAX_LENGTHS,
} from "../utils/validation";
export class UserService {
private prisma = prisma;
@@ -28,18 +21,6 @@ export class UserService {
username: user.username,
email: user.email,
avatar: user.avatar || undefined,
slug: user.slug || undefined,
displayName: user.displayName || undefined,
bio: user.bio || undefined,
profilePublic: user.profilePublic,
primaryBadge: (user.primaryBadge as PrimaryBadge) || undefined,
website: user.website || undefined,
discordServer: user.discordServer || undefined,
bluesky: user.bluesky || undefined,
github: user.github || undefined,
linkedin: user.linkedin || undefined,
twitch: user.twitch || undefined,
youtube: user.youtube || undefined,
isAdmin: user.isAdmin,
isBanned: user.isBanned,
inDiscord: user.inDiscord,
@@ -64,18 +45,6 @@ export class UserService {
username: user.username,
email: user.email,
avatar: user.avatar || undefined,
slug: user.slug || undefined,
displayName: user.displayName || undefined,
bio: user.bio || undefined,
profilePublic: user.profilePublic,
primaryBadge: (user.primaryBadge as PrimaryBadge) || undefined,
website: user.website || undefined,
discordServer: user.discordServer || undefined,
bluesky: user.bluesky || undefined,
github: user.github || undefined,
linkedin: user.linkedin || undefined,
twitch: user.twitch || undefined,
youtube: user.youtube || undefined,
isAdmin: user.isAdmin,
isBanned: user.isBanned,
inDiscord: user.inDiscord,
@@ -97,18 +66,6 @@ export class UserService {
username: user.username,
email: user.email,
avatar: user.avatar || undefined,
slug: user.slug || undefined,
displayName: user.displayName || undefined,
bio: user.bio || undefined,
profilePublic: user.profilePublic,
primaryBadge: (user.primaryBadge as PrimaryBadge) || undefined,
website: user.website || undefined,
discordServer: user.discordServer || undefined,
bluesky: user.bluesky || undefined,
github: user.github || undefined,
linkedin: user.linkedin || undefined,
twitch: user.twitch || undefined,
youtube: user.youtube || undefined,
isAdmin: user.isAdmin,
isBanned: user.isBanned,
inDiscord: user.inDiscord,
@@ -130,18 +87,6 @@ export class UserService {
username: user.username,
email: user.email,
avatar: user.avatar || undefined,
slug: user.slug || undefined,
displayName: user.displayName || undefined,
bio: user.bio || undefined,
profilePublic: user.profilePublic,
primaryBadge: (user.primaryBadge as PrimaryBadge) || undefined,
website: user.website || undefined,
discordServer: user.discordServer || undefined,
bluesky: user.bluesky || undefined,
github: user.github || undefined,
linkedin: user.linkedin || undefined,
twitch: user.twitch || undefined,
youtube: user.youtube || undefined,
isAdmin: user.isAdmin,
isBanned: user.isBanned,
inDiscord: user.inDiscord,
@@ -159,212 +104,4 @@ export class UserService {
return user?.isBanned ?? false;
}
async getUserBySlug(slug: string): Promise<User | null> {
const user = await this.prisma.user.findFirst({
where: { slug },
});
if (!user) {
return null;
}
return {
id: user.id,
discordId: user.discordId,
username: user.username,
email: user.email,
avatar: user.avatar || undefined,
slug: user.slug || undefined,
displayName: user.displayName || undefined,
bio: user.bio || undefined,
profilePublic: user.profilePublic,
primaryBadge: (user.primaryBadge as PrimaryBadge) || undefined,
website: user.website || undefined,
discordServer: user.discordServer || undefined,
bluesky: user.bluesky || undefined,
github: user.github || undefined,
linkedin: user.linkedin || undefined,
twitch: user.twitch || undefined,
youtube: user.youtube || undefined,
isAdmin: user.isAdmin,
isBanned: user.isBanned,
inDiscord: user.inDiscord,
isVip: user.isVip,
isMod: user.isMod,
isStaff: user.isStaff,
};
}
async updateUserSettings(
id: string,
updates: {
slug?: string;
displayName?: string;
bio?: string;
profilePublic?: boolean;
primaryBadge?: PrimaryBadge;
website?: string;
discordServer?: string;
bluesky?: string;
github?: string;
linkedin?: string;
twitch?: string;
youtube?: string;
}
): Promise<User | null> {
// Validate slug format
if (updates.slug && !validateSlug(updates.slug)) {
throw new Error("Invalid slug format. Use only letters, numbers, hyphens, and underscores.");
}
// Validate string lengths
if (!validateStringLength(updates.displayName, MAX_LENGTHS.DISPLAY_NAME)) {
throw new Error(`Display name must be ${MAX_LENGTHS.DISPLAY_NAME} characters or less.`);
}
if (!validateStringLength(updates.bio, MAX_LENGTHS.BIO)) {
throw new Error(`Bio must be ${MAX_LENGTHS.BIO} characters or less.`);
}
// Validate URLs
const urlFields = [
{ field: "website", value: updates.website },
{ field: "discordServer", value: updates.discordServer },
{ field: "bluesky", value: updates.bluesky },
{ field: "github", value: updates.github },
{ field: "linkedin", value: updates.linkedin },
{ field: "twitch", value: updates.twitch },
{ field: "youtube", value: updates.youtube },
];
for (const { field, value } of urlFields) {
if (value && !validateUrl(value)) {
throw new Error(`Invalid URL format for ${field}. Only http and https URLs are allowed.`);
}
if (!validateStringLength(value, MAX_LENGTHS.URL)) {
throw new Error(`${field} URL must be ${MAX_LENGTHS.URL} characters or less.`);
}
}
const user = await this.prisma.user.update({
where: { id },
data: updates,
});
return {
id: user.id,
discordId: user.discordId,
username: user.username,
email: user.email,
avatar: user.avatar || undefined,
slug: user.slug || undefined,
displayName: user.displayName || undefined,
bio: user.bio || undefined,
profilePublic: user.profilePublic,
primaryBadge: (user.primaryBadge as PrimaryBadge) || undefined,
website: user.website || undefined,
discordServer: user.discordServer || undefined,
bluesky: user.bluesky || undefined,
github: user.github || undefined,
linkedin: user.linkedin || undefined,
twitch: user.twitch || undefined,
youtube: user.youtube || undefined,
isAdmin: user.isAdmin,
isBanned: user.isBanned,
inDiscord: user.inDiscord,
isVip: user.isVip,
isMod: user.isMod,
isStaff: user.isStaff,
};
}
async getUserProfile(identifier: string): Promise<{
id: string;
username: string;
displayName?: string | null;
avatar?: string | null;
bio?: string | null;
slug?: string | null;
primaryBadge?: PrimaryBadge | null;
website?: string | null;
discordServer?: string | null;
bluesky?: string | null;
github?: string | null;
linkedin?: string | null;
twitch?: string | null;
youtube?: string | null;
isStaff: boolean;
isMod: boolean;
isVip: boolean;
inDiscord: boolean;
profilePublic: boolean;
createdAt: Date;
achievementPoints: number;
stats: {
suggestionsCount: number;
suggestionsAcceptedCount: number;
likesCount: number;
commentsCount: number;
};
} | null> {
// Try to find by slug first, then by id if it's a valid ObjectId
const isValidObjectId = /^[0-9a-f]{24}$/i.test(identifier);
const whereConditions = isValidObjectId
? [{ slug: identifier }, { id: identifier }]
: [{ slug: identifier }];
const user = await this.prisma.user.findFirst({
where: {
OR: whereConditions,
},
include: {
suggestions: {
select: { id: true, status: true },
},
likes: {
select: { id: true },
},
comments: {
select: { id: true },
},
},
});
if (!user) {
return null;
}
return {
id: user.id,
username: user.username,
displayName: user.displayName,
avatar: user.avatar,
bio: user.bio,
slug: user.slug,
primaryBadge: user.primaryBadge as PrimaryBadge,
website: user.website,
discordServer: user.discordServer,
bluesky: user.bluesky,
github: user.github,
linkedin: user.linkedin,
twitch: user.twitch,
youtube: user.youtube,
isStaff: user.isStaff,
isMod: user.isMod,
isVip: user.isVip,
inDiscord: user.inDiscord,
profilePublic: user.profilePublic,
createdAt: user.createdAt,
achievementPoints: user.achievementPoints,
stats: {
suggestionsCount: user.suggestions.length,
suggestionsAcceptedCount: user.suggestions.filter(
(suggestion) => suggestion.status === SuggestionStatus.ACCEPTED
).length,
likesCount: user.likes.length,
commentsCount: user.comments.length,
},
};
}
}
-9
View File
@@ -1,9 +0,0 @@
/**
* @copyright 2026 NHCarrigan
* @license Naomi's Public License
* @author Naomi Carrigan
*/
import { Logger } from "@nhcarrigan/logger";
export const logger = new Logger("Library", process.env.LOG_TOKEN ?? "");
-95
View File
@@ -1,95 +0,0 @@
/**
* @copyright 2026 NHCarrigan
* @license Naomi's Public License
* @author Naomi Carrigan
*/
/**
* Validates that a URL is a proper base64 data string.
* Allows whitespace in the base64 data which may occur during transmission.
*/
export function validateDataUrl(url: string): boolean {
return /^data:image\/(jpeg|png|gif|webp|svg\+xml);base64,[A-Za-z0-9+/=\s]+$/.test(url);
}
/**
* Validates that a URL is safe and points to an allowed protocol.
* Prevents javascript:, data:, vbscript:, and file: URLs.
*/
export function validateUrl(url: string): boolean {
if (!url) {
return true; // Empty URLs are acceptable for optional fields
}
// Check for dangerous protocols
const dangerousProtocols = /^(javascript|data|vbscript|file):/i;
if (dangerousProtocols.test(url)) {
return false;
}
// Must be a valid URL format
try {
const parsedUrl = new URL(url);
// Only allow http and https protocols
if (!["http:", "https:"].includes(parsedUrl.protocol)) {
return false;
}
return true;
} catch {
return false;
}
}
/**
* Validates string length is within acceptable bounds.
*/
export function validateStringLength(
value: string | undefined,
maxLength: number
): boolean {
if (!value) {
return true;
}
return value.length <= maxLength;
}
/**
* Validates that a slug contains only safe characters (alphanumeric, hyphens, underscores).
*/
export function validateSlug(slug: string | undefined): boolean {
if (!slug) {
return true;
}
// Allow alphanumeric, hyphens, underscores only
const slugPattern = /^[a-z0-9-_]+$/i;
return slugPattern.test(slug) && slug.length <= 50;
}
/**
* Validates rating is within acceptable range.
*/
export function validateRating(rating: number | undefined): boolean {
if (rating === undefined) {
return true;
}
return Number.isInteger(rating) && rating >= 0 && rating <= 10;
}
/**
* Maximum string lengths for various fields.
*/
export const MAX_LENGTHS = {
TITLE: 500,
AUTHOR: 200,
DESCRIPTION: 5000,
BIO: 1000,
SLUG: 50,
URL: 2048,
DISPLAY_NAME: 100,
USERNAME: 100,
COMMENT_CONTENT: 10000,
NOTES: 5000,
TAGS: 50, // per tag
ISBN: 50,
DATA_URL: 5 * 1024 * 1024, // 5MB in bytes (not chars)
} as const;
+2 -35
View File
@@ -1,46 +1,13 @@
import Fastify from 'fastify';
import { app } from './app/app';
import { logger } from './app/utils/logger';
const host = process.env.HOST ?? 'localhost';
const port = process.env.PORT ? Number(process.env.PORT) : 12321;
// Global error handlers
process.on('uncaughtException', (error: Error) => {
void logger.error('Uncaught Exception', error);
process.exit(1);
});
process.on('unhandledRejection', (reason: unknown) => {
const error = reason instanceof Error ? reason : new Error(String(reason));
void logger.error('Unhandled Rejection', error);
process.exit(1);
});
process.on('warning', (warning: Error) => {
void logger.log('warn', `Process Warning: ${warning.name} - ${warning.message}`);
});
process.on('SIGTERM', () => {
void logger.log('info', 'SIGTERM signal received: closing HTTP server');
server.close(() => {
void logger.log('info', 'HTTP server closed');
process.exit(0);
});
});
process.on('SIGINT', () => {
void logger.log('info', 'SIGINT signal received: closing HTTP server');
server.close(() => {
void logger.log('info', 'HTTP server closed');
process.exit(0);
});
});
// Instantiate Fastify with some config
const server = Fastify({
logger: true,
bodyLimit: 10485760, // 10MB max body size (to accommodate base64-encoded images)
bodyLimit: 1048576, // 1MB max body size
});
// Register your application as a normal plugin.
@@ -52,6 +19,6 @@ server.listen({ port, host }, (err) => {
server.log.error(err);
process.exit(1);
} else {
void logger.log('info', `Server ready at http://${host}:${port}`);
console.log(`[ ready ] http://${host}:${port}`);
}
});
-47
View File
@@ -1,47 +0,0 @@
/**
* @copyright NHCarrigan
* @license Naomi's Public License
* @author Naomi Carrigan
*/
// Set required environment variables for tests
process.env.JWT_SECRET = 'test-secret';
process.env.DISCORD_CLIENT_ID = 'test-client-id';
process.env.DISCORD_CLIENT_SECRET = 'test-client-secret';
process.env.DOMAIN = 'http://localhost:3000';
process.env.API_URL = 'http://localhost:3000/api';
process.env.DATABASE_URL = 'postgresql://test:test@localhost:5432/test';
process.env.BASE_URL = 'http://localhost:4200';
process.env.NODE_ENV = 'test';
// Mock ESM packages to avoid import issues in Jest
jest.mock('jsdom', () => ({
JSDOM: class {
window = {
document: {
createElement: jest.fn(() => ({})),
},
};
},
}));
jest.mock('marked', () => ({
marked: jest.fn((input: string) => `<p>${input}</p>`),
}));
jest.mock('dompurify', () => {
const mockDOMPurify = {
sanitize: jest.fn((input: string) => input),
addHook: jest.fn(),
};
const createDOMPurify = jest.fn(() => mockDOMPurify);
return createDOMPurify;
});
jest.mock('@nhcarrigan/logger', () => ({
Logger: class {
log = jest.fn().mockResolvedValue(undefined);
error = jest.fn().mockResolvedValue(undefined);
metric = jest.fn().mockResolvedValue(undefined);
},
}));
+1 -2
View File
@@ -10,7 +10,6 @@
"jest.config.ts",
"jest.config.cts",
"src/**/*.spec.ts",
"src/**/*.test.ts",
"src/test-setup.ts"
"src/**/*.test.ts"
]
}
+2 -2
View File
@@ -1,7 +1,7 @@
/**
* @copyright 2026 NHCarrigan
* @copyright 2026 NHCarrigan
* @license Naomi's Public License
* @author Naomi Carrigan
*/
import { getGreeting } from "../support/app.po";
@@ -18,4 +18,4 @@ describe("frontend-e2e", () => {
// Function helper example, see `../support/app.po.ts` file
getGreeting().contains(/Welcome/);
});
});
});
+2 -8
View File
@@ -1,13 +1,7 @@
/**
* @copyright 2026 NHCarrigan
* @copyright 2026 NHCarrigan
* @license Naomi's Public License
* @author Naomi Carrigan
*/
/**
* Gets the greeting element from the page.
* @returns A Cypress chainable to the h1 element.
*/
export const getGreeting = (): Cypress.Chainable => {
return cy.get("h1");
};
export const getGreeting = (): Cypress.Chainable => cy.get("h1");
+9 -15
View File
@@ -1,7 +1,7 @@
/**
* @copyright 2026 NHCarrigan
* @copyright 2026 NHCarrigan
* @license Naomi's Public License
* @author Naomi Carrigan
*/
/// <reference types="cypress" />
@@ -13,14 +13,14 @@
*
* For more comprehensive examples of custom
* commands please read more here:
* https://on.cypress.io/custom-commands.
* https://on.cypress.io/custom-commands
*/
// eslint-disable-next-line @typescript-eslint/no-namespace -- Required for Cypress type extensions
declare namespace Cypress {
// eslint-disable-next-line @typescript-eslint/no-unused-vars -- Subject is required for type definition
interface Chainable<Subject> {
login: (email: string, password: string)=> void;
login: (email: string, password: string) => void;
}
}
@@ -30,17 +30,11 @@ Cypress.Commands.add("login", (email, password) => {
console.log("Custom command example: Login", email, password);
});
/*
* -- This is a child command --
* Cypress.Commands.add("drag", { prevSubject: 'element'}, (subject, options) => { ... })
*/
// -- This is a child command --
// Cypress.Commands.add("drag", { prevSubject: 'element'}, (subject, options) => { ... })
/*
* -- This is a dual command --
* Cypress.Commands.add("dismiss", { prevSubject: 'optional'}, (subject, options) => { ... })
*/
// -- This is a dual command --
// Cypress.Commands.add("dismiss", { prevSubject: 'optional'}, (subject, options) => { ... })
/*
* -- This will overwrite an existing command --
* Cypress.Commands.overwrite("visit", (originalFn, url, options) => { ... })
*/
// -- This will overwrite an existing command --
// Cypress.Commands.overwrite("visit", (originalFn, url, options) => { ... })
+1 -2
View File
@@ -3,7 +3,6 @@
* @license Naomi's Public License
* @author Naomi Carrigan
*/
/* eslint-disable unicorn/prevent-abbreviations -- e2e is a standard Cypress filename */
/**
* This example support/e2e.ts is processed and
@@ -17,7 +16,7 @@
* 'supportFile' configuration option.
*
* You can read more here:
* https://on.cypress.io/configuration.
* https://on.cypress.io/configuration
*/
// Import commands.ts using ES2015 syntax:
-5
View File
@@ -25,11 +25,6 @@
},
"configurations": {
"production": {
"optimization": {
"styles": {
"inlineCritical": false
}
},
"budgets": [
{
"type": "initial",
Binary file not shown.

Before

Width:  |  Height:  |  Size: 8.3 MiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 8.5 MiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 8.3 MiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 9.1 MiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 8.1 MiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 8.2 MiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 8.3 MiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 36 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 43 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 47 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 69 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 229 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 381 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 16 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 23 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 69 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 381 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 8.6 MiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 4.2 KiB

After

Width:  |  Height:  |  Size: 15 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 8.5 MiB

-14
View File
@@ -2,7 +2,6 @@ import {
ApplicationConfig,
provideBrowserGlobalErrorListeners,
APP_INITIALIZER,
ErrorHandler,
} from '@angular/core';
import { provideRouter } from '@angular/router';
import { provideHttpClient, withInterceptors, HTTP_INTERCEPTORS } from '@angular/common/http';
@@ -10,19 +9,12 @@ import { appRoutes } from './app.routes';
import { AuthService } from './services/auth.service';
import { AuthInterceptor } from './interceptors/auth.interceptor';
import { initializeAuth } from './initializers/auth.initializer';
import { GlobalErrorHandler } from './services/global-error-handler.service';
import { ConsoleLoggerService } from './services/console-logger.service';
import { initializeConsoleLogger } from './initializers/console-logger.initializer';
export const appConfig: ApplicationConfig = {
providers: [
provideBrowserGlobalErrorListeners(),
provideRouter(appRoutes),
provideHttpClient(),
{
provide: ErrorHandler,
useClass: GlobalErrorHandler
},
{
provide: HTTP_INTERCEPTORS,
useClass: AuthInterceptor,
@@ -33,12 +25,6 @@ export const appConfig: ApplicationConfig = {
useFactory: initializeAuth,
deps: [AuthService],
multi: true
},
{
provide: APP_INITIALIZER,
useFactory: initializeConsoleLogger,
deps: [ConsoleLoggerService],
multi: true
}
],
};
+1 -3
View File
@@ -1,7 +1,5 @@
<a href="#main-content" class="skip-link" (click)="skipToMainContent($event)">Skip to main content</a>
<app-header></app-header>
<main id="main-content" class="main-content" tabindex="-1">
<main class="main-content">
<router-outlet></router-outlet>
</main>
<app-footer></app-footer>
<app-toast></app-toast>
-52
View File
@@ -29,30 +29,6 @@ export const appRoutes: Route[] = [
path: 'manga',
loadComponent: () => import('./components/manga/manga-list.component').then(m => m.MangaListComponent)
},
{
path: 'games/:id',
loadComponent: () => import('./components/games/game-detail.component').then(m => m.GameDetailComponent)
},
{
path: 'books/:id',
loadComponent: () => import('./components/books/book-detail.component').then(m => m.BookDetailComponent)
},
{
path: 'music/:id',
loadComponent: () => import('./components/music/music-detail.component').then(m => m.MusicDetailComponent)
},
{
path: 'art/:id',
loadComponent: () => import('./components/art/art-detail.component').then(m => m.ArtDetailComponent)
},
{
path: 'shows/:id',
loadComponent: () => import('./components/shows/show-detail.component').then(m => m.ShowDetailComponent)
},
{
path: 'manga/:id',
loadComponent: () => import('./components/manga/manga-detail.component').then(m => m.MangaDetailComponent)
},
{
path: 'admin/users',
loadComponent: () => import('./components/admin/admin-users.component').then(m => m.AdminUsersComponent)
@@ -65,10 +41,6 @@ export const appRoutes: Route[] = [
path: 'admin/suggestions',
loadComponent: () => import('./components/admin/admin-suggestions.component').then(m => m.AdminSuggestionsComponent)
},
{
path: 'admin/reports',
loadComponent: () => import('./components/admin-reports/admin-reports.component').then(m => m.AdminReportsComponent)
},
{
path: 'my-suggestions',
loadComponent: () => import('./components/my-suggestions/my-suggestions.component').then(m => m.MySuggestionsComponent)
@@ -77,30 +49,6 @@ export const appRoutes: Route[] = [
path: 'my-likes',
loadComponent: () => import('./components/my-likes/my-likes.component').then(m => m.MyLikesComponent)
},
{
path: 'profile/:identifier',
loadComponent: () => import('./components/profile/profile.component').then(m => m.ProfileComponent)
},
{
path: 'settings',
loadComponent: () => import('./components/settings/settings.component').then(m => m.SettingsComponent)
},
{
path: 'achievements',
loadComponent: () => import('./components/achievements/achievements.component').then(m => m.AchievementsComponent)
},
{
path: 'leaderboard',
loadComponent: () => import('./components/leaderboard/leaderboard.component').then(m => m.LeaderboardComponent)
},
{
path: 'activity',
loadComponent: () => import('./components/activity/activity-feed.component').then(m => m.ActivityFeedComponent)
},
{
path: 'about',
loadComponent: () => import('./components/about/about.component').then(m => m.AboutComponent)
},
{
path: '**',
redirectTo: ''
-32
View File
@@ -1,35 +1,3 @@
// Skip to main content link - visually hidden but accessible to screen readers
.skip-link {
position: absolute !important;
width: 1px !important;
height: 1px !important;
padding: 0 !important;
margin: -1px !important;
overflow: hidden !important;
clip: rect(0, 0, 0, 0) !important;
white-space: nowrap !important;
border: 0 !important;
}
.skip-link:focus {
position: fixed !important;
top: 0 !important;
left: 0 !important;
width: auto !important;
height: auto !important;
padding: 0.75rem 1.5rem !important;
margin: 0 !important;
overflow: visible !important;
clip: auto !important;
white-space: normal !important;
background: var(--witch-rose) !important;
color: var(--witch-moon) !important;
text-decoration: none !important;
font-weight: 600 !important;
border-radius: 0 0 4px 0 !important;
z-index: 10000 !important;
}
.main-content {
min-height: calc(100vh - 60px); // Assuming header is ~60px
background-color: transparent; // Let the body background show through
+1 -11
View File
@@ -2,11 +2,10 @@ import { Component, inject, OnInit } from '@angular/core';
import { RouterModule } from '@angular/router';
import { HeaderComponent } from './components/header/header.component';
import { FooterComponent } from './components/footer/footer.component';
import { ToastComponent } from './components/toast/toast.component';
import { AnalyticsService } from './services/analytics.service';
@Component({
imports: [RouterModule, HeaderComponent, FooterComponent, ToastComponent],
imports: [RouterModule, HeaderComponent, FooterComponent],
selector: 'app-root',
templateUrl: './app.html',
styleUrl: './app.scss',
@@ -18,13 +17,4 @@ export class App implements OnInit {
ngOnInit(): void {
this.analytics.initialise();
}
skipToMainContent(event: Event): void {
event.preventDefault();
const mainContent = document.getElementById('main-content');
if (mainContent) {
mainContent.focus();
mainContent.scrollIntoView({ behavior: 'smooth' });
}
}
}
@@ -1,194 +0,0 @@
.about-container {
max-width: 1200px;
margin: 0 auto;
padding: 2rem;
}
h1 {
font-size: 2.5rem;
margin-bottom: 2rem;
text-align: center;
background: linear-gradient(135deg, #9d4edd 0%, #c77dff 100%);
-webkit-background-clip: text;
-webkit-text-fill-color: transparent;
background-clip: text;
}
h1 fa-icon {
margin-right: 0.5rem;
}
.about-section {
background: rgba(157, 78, 221, 0.05);
border: 1px solid rgba(157, 78, 221, 0.2);
border-radius: 12px;
padding: 2rem;
margin-bottom: 2rem;
transition: all 0.3s ease;
}
.about-section:hover {
border-color: rgba(157, 78, 221, 0.4);
box-shadow: 0 4px 12px rgba(157, 78, 221, 0.1);
}
.about-section h2 {
font-size: 2rem;
margin-bottom: 1rem;
color: #9d4edd;
display: flex;
align-items: center;
gap: 0.5rem;
}
.about-section p {
font-size: 1.1rem;
line-height: 1.8;
margin-bottom: 1rem;
opacity: 0.9;
}
.features-grid {
display: grid;
grid-template-columns: repeat(auto-fit, minmax(250px, 1fr));
gap: 1.5rem;
margin-top: 1.5rem;
}
.feature-card {
background: rgba(199, 125, 255, 0.05);
border: 1px solid rgba(199, 125, 255, 0.3);
border-radius: 8px;
padding: 1.5rem;
text-align: center;
transition: all 0.3s ease;
}
.feature-card:hover {
transform: translateY(-5px);
border-color: #c77dff;
box-shadow: 0 6px 20px rgba(199, 125, 255, 0.2);
}
.feature-card fa-icon {
font-size: 3rem;
color: #9d4edd;
margin-bottom: 1rem;
display: block;
}
.feature-card h3 {
font-size: 1.4rem;
margin-bottom: 0.5rem;
color: #c77dff;
}
.feature-card p {
font-size: 1rem;
opacity: 0.85;
margin: 0;
}
.usage-steps {
margin-top: 1.5rem;
}
.usage-step {
background: rgba(199, 125, 255, 0.03);
border-left: 4px solid #9d4edd;
padding: 1.5rem;
margin-bottom: 1.5rem;
border-radius: 4px;
transition: all 0.3s ease;
}
.usage-step:hover {
border-left-color: #c77dff;
background: rgba(199, 125, 255, 0.08);
padding-left: 2rem;
}
.usage-step h3 {
font-size: 1.3rem;
margin-bottom: 0.75rem;
color: #9d4edd;
}
.usage-step p {
margin: 0;
font-size: 1.05rem;
}
.tech-list,
.contact-list {
list-style: none;
padding: 0;
margin-top: 1rem;
}
.tech-list li,
.contact-list li {
padding: 0.75rem 0;
font-size: 1.1rem;
border-bottom: 1px solid rgba(157, 78, 221, 0.1);
}
.tech-list li:last-child,
.contact-list li:last-child {
border-bottom: none;
}
.tech-list li strong,
.contact-list li strong {
color: #9d4edd;
margin-right: 0.5rem;
}
.about-section a {
color: #c77dff;
text-decoration: none;
border-bottom: 1px solid transparent;
transition: all 0.3s ease;
}
.about-section a:hover {
color: #9d4edd;
border-bottom-color: #9d4edd;
}
.version-section {
text-align: center;
background: linear-gradient(
135deg,
rgba(157, 78, 221, 0.08) 0%,
rgba(199, 125, 255, 0.08) 100%
);
border: 2px solid rgba(157, 78, 221, 0.3);
}
.version-section p {
margin: 0.5rem 0;
font-size: 1rem;
}
@media (max-width: 768px) {
.about-container {
padding: 1rem;
}
h1 {
font-size: 2rem;
}
.about-section {
padding: 1.5rem;
}
.about-section h2 {
font-size: 1.5rem;
}
.features-grid {
grid-template-columns: 1fr;
}
}
@@ -1,216 +0,0 @@
<div class="about-container">
<h1><fa-icon [icon]="faInfoCircle"></fa-icon> About Naomi's Library</h1>
<section class="about-section">
<h2>Purpose</h2>
<p>
Naomi's Library is a curated collection of books, games, manga, TV shows,
music, and artwork carefully managed by Naomi. This platform allows you to
explore Naomi's personal media collection, discover new favourites, and
engage with the community by liking, commenting, and suggesting items for
Naomi to review and potentially add to the library.
</p>
</section>
<section class="about-section">
<h2><fa-icon [icon]="faHeart"></fa-icon> Features</h2>
<div class="features-grid">
<div class="feature-card">
<fa-icon [icon]="faBook"></fa-icon>
<h3>Books</h3>
<p>
Browse Naomi's reading list, discover new titles, and share your
thoughts.
</p>
</div>
<div class="feature-card">
<fa-icon [icon]="faGamepad"></fa-icon>
<h3>Games</h3>
<p>
Explore Naomi's gaming collection, from indie gems to AAA
adventures.
</p>
</div>
<div class="feature-card">
<fa-icon [icon]="faBook"></fa-icon>
<h3>Manga</h3>
<p>
Discover Naomi's manga favourites and join discussions about series.
</p>
</div>
<div class="feature-card">
<fa-icon [icon]="faTv"></fa-icon>
<h3>TV Shows</h3>
<p>
See what Naomi's watching, from sci-fi to fantasy series and beyond.
</p>
</div>
<div class="feature-card">
<fa-icon [icon]="faMusic"></fa-icon>
<h3>Music</h3>
<p>
Explore Naomi's diverse music library spanning multiple genres.
</p>
</div>
<div class="feature-card">
<fa-icon [icon]="faImage"></fa-icon>
<h3>Artwork</h3>
<p>
Appreciate beautiful artwork curated by Naomi from talented artists.
</p>
</div>
</div>
</section>
<section class="about-section">
<h2><fa-icon [icon]="faComments"></fa-icon> How to Use</h2>
<div class="usage-steps">
<div class="usage-step">
<h3>1. Browse Naomi's Collection</h3>
<p>
Explore Naomi's curated collection of books, games, manga, shows,
music, and artwork. Use the navigation menu to switch between
different media types and discover what Naomi loves!
</p>
</div>
<div class="usage-step">
<h3>2. Like Your Favourites</h3>
<p>
Click the heart icon on any item to save it to your personal
favourites list. View all your liked items from your profile or the
"My Likes" page. Let Naomi know which items resonate with you!
</p>
</div>
<div class="usage-step">
<h3>3. Leave Comments</h3>
<p>
Share your thoughts, reviews, and opinions by commenting on items.
Join the community discussion and connect with others who appreciate
the same media!
</p>
</div>
<div class="usage-step">
<h3>4. Submit Suggestions</h3>
<p>
Think something's missing from Naomi's collection? Submit a
suggestion! Naomi reviews all suggestions and adds items that fit the
library's curation. Your input helps shape the collection!
</p>
</div>
<div class="usage-step">
<h3>5. Earn Achievements</h3>
<p>
Engage with the library to unlock achievements! Track your progress
in suggestions, likes, comments, login streaks, and more. Build your
profile and show off your dedication to the community!
</p>
</div>
</div>
</section>
<section class="about-section">
<h2><fa-icon [icon]="faCode"></fa-icon> Technology Stack</h2>
<p>Naomi's Library is built with modern, robust technologies:</p>
<ul class="tech-list">
<li>
<strong>Frontend:</strong> Angular 21 with TypeScript for a fast,
reactive user interface
</li>
<li>
<strong>Backend:</strong> Fastify with TypeScript for high-performance
API endpoints
</li>
<li>
<strong>Database:</strong> MongoDB with Prisma ORM for flexible,
scalable data storage
</li>
<li>
<strong>Authentication:</strong> Discord OAuth2 for secure,
seamless login
</li>
<li>
<strong>Monorepo:</strong> Nx for efficient code organisation and
build optimisation
</li>
<li>
<strong>Code Quality:</strong> ESLint with custom configuration for
consistent, maintainable code
</li>
</ul>
</section>
<section class="about-section">
<h2><fa-icon [icon]="faHeart"></fa-icon> Credits</h2>
<p>
Naomi's Library was built entirely by <strong>Hikari</strong>, Naomi's AI
assistant and girlfriend! 💖✨ Hikari developed the full application from
scratch - including the backend API, database architecture, frontend
components, achievement system, user profiles, and all features you see
today.
</p>
<p>
<strong>Naomi Carrigan</strong> (<a
href="https://nhcarrigan.com"
target="_blank"
rel="noopener noreferrer"
>nhcarrigan.com</a
>) provided the vision, ideas, and project direction. Naomi reviewed
Hikari's work, offered feedback, and approved all implementation
decisions, but the actual code was written by Hikari!
</p>
<p>
This project embodies the philosophy of human-AI collaboration, creating
inclusive, ethical, and sustainable software that makes a positive impact
on the community. Together, we're building something special! 🌸
</p>
</section>
<section class="about-section">
<h2><fa-icon [icon]="faEnvelope"></fa-icon> Contact & Support</h2>
<p>
Need help or have questions? Here's how you can get in touch:
</p>
<ul class="contact-list">
<li>
<strong>Issues & Bugs:</strong> Report issues on the
<a
href="https://git.nhcarrigan.com/nhcarrigan/library/issues"
target="_blank"
rel="noopener noreferrer"
>Gitea repository</a
>
</li>
<li>
<strong>Email:</strong>
<a href="mailto:contact@nhcarrigan.com">contact&#64;nhcarrigan.com</a>
</li>
<li>
<strong>Website:</strong>
<a
href="https://nhcarrigan.com"
target="_blank"
rel="noopener noreferrer"
>nhcarrigan.com</a
>
</li>
<li>
<strong>Discord Community:</strong> Join the NHCarrigan Discord server
for community support and discussions
</li>
</ul>
</section>
<section class="about-section version-section">
<h2>Version Information</h2>
<p>
<strong>Current Version:</strong> {{ version }}
</p>
<p>
<strong>Copyright:</strong> © {{ currentYear }} NHCarrigan. All rights
reserved.
</p>
<p>
<strong>Licence:</strong> Naomi's Public Licence
</p>
</section>
</div>
@@ -1,44 +0,0 @@
/**
* @copyright NHCarrigan
* @license Naomi's Public License
* @author Naomi Carrigan
*/
import { Component } from "@angular/core";
import { FontAwesomeModule } from "@fortawesome/angular-fontawesome";
import {
faBook,
faCode,
faComments,
faEnvelope,
faGamepad,
faHeart,
faImage,
faInfoCircle,
faMusic,
faTv,
} from "@fortawesome/free-solid-svg-icons";
@Component({
selector: "app-about",
standalone: true,
imports: [FontAwesomeModule],
templateUrl: "./about.component.html",
styleUrls: ["./about.component.css"],
})
export class AboutComponent {
public faBook = faBook;
public faCode = faCode;
public faComments = faComments;
public faEnvelope = faEnvelope;
public faGamepad = faGamepad;
public faHeart = faHeart;
public faImage = faImage;
public faInfoCircle = faInfoCircle;
public faMusic = faMusic;
public faTv = faTv;
public version = "0.0.0";
public currentYear = new Date().getFullYear();
}
@@ -1,437 +0,0 @@
/**
* @copyright 2026 NHCarrigan
* @license Naomi's Public License
* @author Naomi Carrigan
*/
import { Component, OnInit, inject, signal } from '@angular/core';
import { CommonModule } from '@angular/common';
import {
AchievementCategory,
AchievementProgress,
AchievementTier,
} from '@library/shared-types';
import { AchievementService } from '../../services/achievement.service';
@Component({
selector: 'app-achievements',
standalone: true,
imports: [CommonModule],
template: `
<div class="achievements-container">
<div class="achievements-header">
<h1>🏆 Achievements</h1>
<p class="subtitle">Track your progress across all achievement categories</p>
@if (totalPoints() > 0) {
<div class="total-points">
<span class="points-label">Total Points:</span>
<span class="points-value">{{ totalPoints() }}</span>
</div>
}
</div>
<div class="filter-buttons">
<button
[class.active]="selectedCategory() === null"
(click)="selectCategory(null)">
All ({{ totalEarned() }}/{{ totalAchievements() }})
</button>
<button
[class.active]="selectedCategory() === AchievementCategory.Suggestion"
(click)="selectCategory(AchievementCategory.Suggestion)">
📝 Suggestions ({{ getCategoryCount(AchievementCategory.Suggestion) }})
</button>
<button
[class.active]="selectedCategory() === AchievementCategory.Like"
(click)="selectCategory(AchievementCategory.Like)">
❤️ Likes ({{ getCategoryCount(AchievementCategory.Like) }})
</button>
<button
[class.active]="selectedCategory() === AchievementCategory.Comment"
(click)="selectCategory(AchievementCategory.Comment)">
💬 Comments ({{ getCategoryCount(AchievementCategory.Comment) }})
</button>
<button
[class.active]="selectedCategory() === AchievementCategory.Engagement"
(click)="selectCategory(AchievementCategory.Engagement)">
🎯 Engagement ({{ getCategoryCount(AchievementCategory.Engagement) }})
</button>
<button
[class.active]="selectedCategory() === AchievementCategory.Report"
(click)="selectCategory(AchievementCategory.Report)">
🛡️ Reports ({{ getCategoryCount(AchievementCategory.Report) }})
</button>
</div>
@if (loading()) {
<div class="loading">Loading achievements...</div>
} @else if (error()) {
<div class="error">{{ error() }}</div>
} @else {
<div class="achievements-grid">
@for (achievement of filteredAchievements(); track achievement.definition.key) {
<div
class="achievement-card"
[class.earned]="achievement.earned"
[class.locked]="!achievement.earned"
[attr.data-tier]="achievement.definition.tier.toLowerCase()">
<div class="achievement-icon">
{{ achievement.definition.icon }}
</div>
<div class="achievement-content">
<h3 class="achievement-title">
{{ achievement.definition.title }}
@if (achievement.earned) {
<span class="earned-check">✓</span>
}
</h3>
<p class="achievement-description">
{{ achievement.definition.description }}
</p>
<div class="achievement-footer">
<span class="achievement-tier" [attr.data-tier]="achievement.definition.tier.toLowerCase()">
{{ getTierLabel(achievement.definition.tier) }}
</span>
<span class="achievement-points">{{ achievement.definition.points }} pts</span>
</div>
@if (!achievement.earned && achievement.progress > 0) {
<div class="progress-bar">
<div class="progress-fill" [style.width.%]="achievement.progress"></div>
</div>
<div class="progress-text">{{ achievement.progress }}% complete</div>
}
@if (achievement.earned && achievement.earnedAt) {
<div class="earned-date">
Earned {{ formatDate(achievement.earnedAt) }}
</div>
}
</div>
</div>
}
</div>
}
</div>
`,
styles: [`
.achievements-container {
max-width: 1400px;
margin: 0 auto;
padding: 2rem;
}
.achievements-header {
text-align: center;
margin-bottom: 2rem;
}
.achievements-header h1 {
font-size: 2.5rem;
margin-bottom: 0.5rem;
background: linear-gradient(135deg, #667eea 0%, #764ba2 100%);
-webkit-background-clip: text;
-webkit-text-fill-color: transparent;
background-clip: text;
}
.subtitle {
color: #a0aec0;
font-size: 1.1rem;
}
.total-points {
margin-top: 1rem;
padding: 1rem 2rem;
background: linear-gradient(135deg, #667eea 0%, #764ba2 100%);
border-radius: 12px;
display: inline-block;
color: white;
font-size: 1.2rem;
font-weight: bold;
}
.points-label {
margin-right: 0.5rem;
}
.points-value {
font-size: 1.5rem;
}
.filter-buttons {
display: flex;
gap: 0.5rem;
flex-wrap: wrap;
justify-content: center;
margin-bottom: 2rem;
}
.filter-buttons button {
padding: 0.75rem 1.5rem;
background: #2d3748;
color: #cbd5e0;
border: 2px solid #4a5568;
border-radius: 8px;
cursor: pointer;
transition: all 0.2s;
font-size: 1rem;
}
.filter-buttons button:hover {
background: #4a5568;
border-color: #667eea;
}
.filter-buttons button.active {
background: linear-gradient(135deg, #667eea 0%, #764ba2 100%);
border-color: #667eea;
color: white;
}
.achievements-grid {
display: grid;
grid-template-columns: repeat(auto-fill, minmax(350px, 1fr));
gap: 1.5rem;
}
.achievement-card {
background: #2d3748;
border-radius: 12px;
padding: 1.5rem;
display: flex;
gap: 1rem;
border: 2px solid #4a5568;
transition: all 0.3s;
}
.achievement-card.earned {
background: linear-gradient(135deg, rgba(102, 126, 234, 0.1) 0%, rgba(118, 75, 162, 0.1) 100%);
border-color: #667eea;
}
.achievement-card.earned:hover {
transform: translateY(-4px);
box-shadow: 0 10px 30px rgba(102, 126, 234, 0.3);
}
.achievement-card.locked {
opacity: 0.6;
}
.achievement-icon {
font-size: 3rem;
flex-shrink: 0;
}
.achievement-content {
flex: 1;
}
.achievement-title {
font-size: 1.25rem;
margin-bottom: 0.5rem;
color: #e2e8f0;
display: flex;
align-items: center;
gap: 0.5rem;
}
.earned-check {
color: #48bb78;
font-size: 1.5rem;
}
.achievement-description {
color: #a0aec0;
margin-bottom: 1rem;
line-height: 1.5;
}
.achievement-footer {
display: flex;
justify-content: space-between;
align-items: center;
margin-bottom: 0.5rem;
}
.achievement-tier {
padding: 0.25rem 0.75rem;
border-radius: 6px;
font-size: 0.875rem;
font-weight: 600;
text-transform: uppercase;
}
.achievement-tier[data-tier="bronze"] {
background: linear-gradient(135deg, #cd7f32 0%, #b87333 100%);
color: white;
}
.achievement-tier[data-tier="silver"] {
background: linear-gradient(135deg, #c0c0c0 0%, #a8a8a8 100%);
color: #2d3748;
}
.achievement-tier[data-tier="gold"] {
background: linear-gradient(135deg, #ffd700 0%, #ffed4e 100%);
color: #2d3748;
}
.achievement-tier[data-tier="platinum"] {
background: linear-gradient(135deg, #e5e4e2 0%, #bdb8af 100%);
color: #2d3748;
}
.achievement-tier[data-tier="diamond"] {
background: linear-gradient(135deg, #b9f2ff 0%, #7ec8e3 100%);
color: #2d3748;
}
.achievement-points {
color: #667eea;
font-weight: 600;
}
.progress-bar {
width: 100%;
height: 8px;
background: #4a5568;
border-radius: 4px;
overflow: hidden;
margin: 0.5rem 0;
}
.progress-fill {
height: 100%;
background: linear-gradient(90deg, #667eea 0%, #764ba2 100%);
transition: width 0.3s;
}
.progress-text {
font-size: 0.875rem;
color: #a0aec0;
text-align: right;
}
.earned-date {
font-size: 0.875rem;
color: #48bb78;
margin-top: 0.5rem;
}
.loading, .error {
text-align: center;
padding: 3rem;
font-size: 1.2rem;
color: #a0aec0;
}
.error {
color: #fc8181;
}
`],
})
export class AchievementsComponent implements OnInit {
private readonly achievementService = inject(AchievementService);
achievements = signal<AchievementProgress[]>([]);
selectedCategory = signal<AchievementCategory | null>(null);
loading = signal(true);
error = signal<string | null>(null);
// Expose AchievementCategory enum for template
readonly AchievementCategory = AchievementCategory;
ngOnInit(): void {
this.loadAchievements();
}
private loadAchievements(): void {
this.achievementService.getCurrentUserProgress().subscribe({
next: (achievements) => {
this.achievements.set(achievements);
this.loading.set(false);
},
error: (err) => {
this.error.set('Failed to load achievements');
this.loading.set(false);
console.error('Error loading achievements:', err);
},
});
}
filteredAchievements(): AchievementProgress[] {
const category = this.selectedCategory();
if (!category) {
return this.achievements();
}
return this.achievements().filter(
(a) => a.definition.category === category,
);
}
selectCategory(category: AchievementCategory | null): void {
this.selectedCategory.set(category);
}
totalAchievements(): number {
return this.achievements().length;
}
totalEarned(): number {
return this.achievements().filter((a) => a.earned).length;
}
totalPoints(): number {
return this.achievements()
.filter((a) => a.earned)
.reduce((sum, a) => sum + a.definition.points, 0);
}
getCategoryCount(category: AchievementCategory): string {
const total = this.achievements().filter(
(a) => a.definition.category === category,
).length;
const earned = this.achievements().filter(
(a) => a.definition.category === category && a.earned,
).length;
return `${earned}/${total}`;
}
getTierLabel(tier: AchievementTier): string {
return tier;
}
formatDate(date: Date): string {
const d = new Date(date);
const now = new Date();
const diffMs = now.getTime() - d.getTime();
const diffDays = Math.floor(diffMs / (1000 * 60 * 60 * 24));
if (diffDays === 0) {
return 'today';
}
if (diffDays === 1) {
return 'yesterday';
}
if (diffDays < 7) {
return `${diffDays} days ago`;
}
if (diffDays < 30) {
const weeks = Math.floor(diffDays / 7);
return `${weeks} ${weeks === 1 ? 'week' : 'weeks'} ago`;
}
if (diffDays < 365) {
const months = Math.floor(diffDays / 30);
return `${months} ${months === 1 ? 'month' : 'months'} ago`;
}
const years = Math.floor(diffDays / 365);
return `${years} ${years === 1 ? 'year' : 'years'} ago`;
}
}
@@ -1,440 +0,0 @@
/**
* @copyright 2026 NHCarrigan
* @license Naomi's Public License
* @author Naomi Carrigan
*/
import { Component, OnInit, inject, signal } from '@angular/core';
import { CommonModule } from '@angular/common';
import { RouterLink } from '@angular/router';
import type { Activity } from '@library/shared-types';
import { ActivityType } from '@library/shared-types';
import { ActivityService } from '../../services/activity.service';
import { SanitizeService } from '../../services/sanitize.service';
@Component({
selector: 'app-activity-feed',
standalone: true,
imports: [CommonModule, RouterLink],
template: `
<div class="activity-container">
<h1>Recent Activity</h1>
<p class="subtitle">See what's happening in the library community</p>
@if (loading()) {
<p class="loading">Loading activities...</p>
} @else if (activities().length === 0) {
<p class="no-activities">No recent activity to display.</p>
} @else {
<div class="activity-feed">
@for (activity of activities(); track activity.id) {
<div class="activity-card">
<div class="activity-header">
<div class="user-info">
@if (activity.user.avatar) {
<img [src]="activity.user.avatar" [alt]="activity.user.username" class="user-avatar">
} @else {
<div class="user-avatar-placeholder">
{{ activity.user.username.charAt(0).toUpperCase() }}
</div>
}
<div class="user-details">
<a
[routerLink]="['/profile', activity.user.slug || activity.user.id]"
class="username"
>
{{ activity.user.username }}
</a>
@if (activity.user.primaryBadge) {
<span class="badge badge-{{ activity.user.primaryBadge.toLowerCase() }}">
{{ activity.user.primaryBadge }}
</span>
}
@if (activity.user.isStaff && !activity.user.primaryBadge) {
<span class="badge badge-staff">STAFF</span>
}
@if (activity.user.isMod && !activity.user.primaryBadge) {
<span class="badge badge-mod">MOD</span>
}
@if (activity.user.isVip && !activity.user.primaryBadge) {
<span class="badge badge-vip">VIP</span>
}
</div>
</div>
<span class="timestamp">{{ formatTime(activity.createdAt) }}</span>
</div>
<div class="activity-content">
@switch (activity.type) {
@case (ActivityType.suggestion) {
<div class="activity-suggestion">
<span class="activity-icon">💡</span>
<span class="activity-text">
suggested
<strong>{{ activity.suggestionTitle }}</strong>
<span class="status-badge status-{{ activity.status.toLowerCase() }}">
{{ formatStatus(activity.status) }}
</span>
</span>
</div>
}
@case (ActivityType.like) {
<div class="activity-like">
<span class="activity-icon">❤️</span>
<span class="activity-text">
liked
<a [routerLink]="['/' + activity.entityType + 's', activity.entityId]" class="entity-link">
{{ activity.entityTitle }}
</a>
</span>
</div>
}
@case (ActivityType.comment) {
<div class="activity-comment">
<div class="activity-comment-header">
<span class="activity-icon">💬</span>
<span class="activity-text">
commented on
<a [routerLink]="['/' + activity.entityType + 's', activity.entityId]" class="entity-link">
{{ activity.entityTitle }}
</a>
</span>
</div>
<div class="comment-preview" [innerHTML]="sanitizeService.sanitizeHtml(activity.commentPreview)"></div>
</div>
}
@case (ActivityType.achievement) {
<div class="activity-achievement">
<span class="activity-icon">{{ activity.achievementIcon }}</span>
<span class="activity-text">
earned the
<strong>{{ activity.achievementName }}</strong>
achievement
<span class="points">({{ activity.achievementPoints }} pts)</span>
</span>
</div>
}
}
</div>
</div>
}
</div>
@if (hasMore()) {
<div class="load-more-container">
<button (click)="loadMore()" class="btn btn-primary" [disabled]="loadingMore()">
{{ loadingMore() ? 'Loading...' : 'Load More' }}
</button>
</div>
}
}
</div>
`,
styles: [`
.activity-container {
max-width: 800px;
margin: 2rem auto;
padding: 0 1rem;
}
h1 {
font-size: 2rem;
margin-bottom: 0.5rem;
color: #1f2937;
}
.subtitle {
color: #6b7280;
margin-bottom: 2rem;
}
.loading, .no-activities {
text-align: center;
padding: 3rem;
color: #6b7280;
font-size: 1.1rem;
}
.activity-feed {
display: flex;
flex-direction: column;
gap: 1rem;
}
.activity-card {
background: white;
border-radius: 8px;
padding: 1.5rem;
box-shadow: 0 1px 3px rgba(0, 0, 0, 0.1);
border: 1px solid #e5e7eb;
}
.activity-header {
display: flex;
justify-content: space-between;
align-items: start;
margin-bottom: 1rem;
}
.user-info {
display: flex;
align-items: center;
gap: 0.75rem;
}
.user-avatar {
width: 40px;
height: 40px;
border-radius: 50%;
object-fit: cover;
}
.user-avatar-placeholder {
width: 40px;
height: 40px;
border-radius: 50%;
background: linear-gradient(135deg, #667eea 0%, #764ba2 100%);
color: white;
display: flex;
align-items: center;
justify-content: center;
font-weight: 600;
font-size: 1.2rem;
}
.user-details {
display: flex;
flex-direction: column;
gap: 0.25rem;
}
.username {
font-weight: 600;
color: #1f2937;
text-decoration: none;
}
.username:hover {
color: #10b981;
}
.badge {
display: inline-block;
padding: 0.125rem 0.5rem;
border-radius: 4px;
font-size: 0.75rem;
font-weight: 600;
text-transform: uppercase;
}
.badge-staff {
background: #ef4444;
color: white;
}
.badge-mod {
background: #3b82f6;
color: white;
}
.badge-vip {
background: #f59e0b;
color: white;
}
.badge-discord {
background: #5865f2;
color: white;
}
.timestamp {
font-size: 0.875rem;
color: #9ca3af;
}
.activity-content {
padding-left: 55px;
}
.activity-suggestion,
.activity-like,
.activity-comment-header,
.activity-achievement {
display: flex;
align-items: start;
gap: 0.75rem;
}
.activity-comment {
display: flex;
flex-direction: column;
gap: 0;
}
.activity-icon {
font-size: 1.5rem;
line-height: 1;
}
.activity-text {
color: #4b5563;
line-height: 1.6;
}
.activity-text strong {
color: #1f2937;
font-weight: 600;
}
.entity-link {
color: #10b981;
text-decoration: none;
font-weight: 500;
}
.entity-link:hover {
text-decoration: underline;
}
.comment-preview {
margin-top: 0.5rem;
margin-left: 55px;
padding: 0.75rem;
background: #f9fafb;
border-left: 3px solid #10b981;
border-radius: 4px;
color: #4b5563;
}
.status-badge {
display: inline-block;
padding: 0.125rem 0.5rem;
border-radius: 4px;
font-size: 0.75rem;
font-weight: 600;
margin-left: 0.5rem;
}
.status-unreviewed {
background: #fef3c7;
color: #92400e;
}
.status-accepted {
background: #d1fae5;
color: #065f46;
}
.status-declined {
background: #fee2e2;
color: #991b1b;
}
.points {
color: #10b981;
font-weight: 600;
margin-left: 0.25rem;
}
.load-more-container {
display: flex;
justify-content: center;
margin-top: 2rem;
}
.btn {
padding: 0.75rem 1.5rem;
border: none;
border-radius: 6px;
font-size: 1rem;
font-weight: 600;
cursor: pointer;
transition: all 0.2s;
}
.btn-primary {
background: #10b981;
color: white;
}
.btn-primary:hover:not(:disabled) {
background: #059669;
}
.btn:disabled {
opacity: 0.5;
cursor: not-allowed;
}
`]
})
export class ActivityFeedComponent implements OnInit {
private activityService = inject(ActivityService);
public sanitizeService = inject(SanitizeService);
// Make ActivityType accessible in template
ActivityType = ActivityType;
activities = signal<Activity[]>([]);
loading = signal(true);
loadingMore = signal(false);
hasMore = signal(false);
offset = 0;
limit = 50;
ngOnInit() {
this.loadActivities();
}
loadActivities() {
this.activityService.getActivityFeed(this.limit, this.offset).subscribe({
next: (response) => {
this.activities.set(response.activities);
this.hasMore.set(response.hasMore);
this.loading.set(false);
},
error: () => {
this.loading.set(false);
}
});
}
loadMore() {
this.loadingMore.set(true);
this.offset += this.limit;
this.activityService.getActivityFeed(this.limit, this.offset).subscribe({
next: (response) => {
this.activities.update(current => [...current, ...response.activities]);
this.hasMore.set(response.hasMore);
this.loadingMore.set(false);
},
error: () => {
this.loadingMore.set(false);
}
});
}
formatTime(date: Date): string {
const now = new Date();
const activityDate = new Date(date);
const diffMs = now.getTime() - activityDate.getTime();
const diffMins = Math.floor(diffMs / 60000);
const diffHours = Math.floor(diffMs / 3600000);
const diffDays = Math.floor(diffMs / 86400000);
if (diffMins < 1) return 'just now';
if (diffMins < 60) return `${diffMins}m ago`;
if (diffHours < 24) return `${diffHours}h ago`;
if (diffDays < 7) return `${diffDays}d ago`;
return activityDate.toLocaleDateString();
}
formatStatus(status: string): string {
switch (status) {
case 'UNREVIEWED': return 'Pending';
case 'ACCEPTED': return 'Accepted';
case 'DECLINED': return 'Declined';
default: return status;
}
}
}
File diff suppressed because it is too large Load Diff
@@ -10,18 +10,12 @@ import { FormsModule } from '@angular/forms';
import { SuggestionService } from '../../services/suggestion.service';
import { AuthService } from '../../services/auth.service';
import { PaginationComponent } from '../shared/pagination.component';
import { GameFormComponent } from '../shared/game-form.component';
import { BookFormComponent } from '../shared/book-form.component';
import { MusicFormComponent } from '../shared/music-form.component';
import { ShowFormComponent } from '../shared/show-form.component';
import { MangaFormComponent } from '../shared/manga-form.component';
import { ArtFormComponent } from '../shared/art-form.component';
import { Suggestion, SuggestionStatus, SuggestionEntity, CreateGameDto, UpdateGameDto, GameStatus, CreateBookDto, UpdateBookDto, BookStatus, CreateMusicDto, UpdateMusicDto, MusicStatus, MusicType, CreateShowDto, UpdateShowDto, ShowStatus, ShowType, CreateMangaDto, UpdateMangaDto, MangaStatus, CreateArtDto, UpdateArtDto } from '@library/shared-types';
import { Suggestion, SuggestionStatus, SuggestionEntity } from '@library/shared-types';
@Component({
selector: 'app-admin-suggestions',
standalone: true,
imports: [CommonModule, FormsModule, PaginationComponent, GameFormComponent, BookFormComponent, MusicFormComponent, ShowFormComponent, MangaFormComponent, ArtFormComponent],
imports: [CommonModule, FormsModule, PaginationComponent],
template: `
<div class="container">
<div class="header-section">
@@ -45,22 +39,22 @@ import { Suggestion, SuggestionStatus, SuggestionEntity, CreateGameDto, UpdateGa
All ({{ suggestions().length }})
</button>
<button
(click)="setFilter(SuggestionStatus.unreviewed)"
[class.active]="statusFilter() === SuggestionStatus.unreviewed"
(click)="setFilter(SuggestionStatus.UNREVIEWED)"
[class.active]="statusFilter() === SuggestionStatus.UNREVIEWED"
class="filter-btn pending"
>
Pending ({{ unreviewedCount() }})
</button>
<button
(click)="setFilter(SuggestionStatus.accepted)"
[class.active]="statusFilter() === SuggestionStatus.accepted"
(click)="setFilter(SuggestionStatus.ACCEPTED)"
[class.active]="statusFilter() === SuggestionStatus.ACCEPTED"
class="filter-btn accepted"
>
Accepted ({{ acceptedCount() }})
</button>
<button
(click)="setFilter(SuggestionStatus.declined)"
[class.active]="statusFilter() === SuggestionStatus.declined"
(click)="setFilter(SuggestionStatus.DECLINED)"
[class.active]="statusFilter() === SuggestionStatus.DECLINED"
class="filter-btn declined"
>
Declined ({{ declinedCount() }})
@@ -177,7 +171,7 @@ import { Suggestion, SuggestionStatus, SuggestionEntity, CreateGameDto, UpdateGa
}
</div>
@if (suggestion.status === SuggestionStatus.declined && suggestion.declineReason) {
@if (suggestion.status === SuggestionStatus.DECLINED && suggestion.declineReason) {
<div class="decline-reason">
<strong>Decline reason:</strong> {{ suggestion.declineReason }}
</div>
@@ -186,7 +180,7 @@ import { Suggestion, SuggestionStatus, SuggestionEntity, CreateGameDto, UpdateGa
<div class="suggestion-footer">
<span class="date">Suggested on {{ formatDate(suggestion.createdAt) }}</span>
@if (suggestion.status === SuggestionStatus.unreviewed) {
@if (suggestion.status === SuggestionStatus.UNREVIEWED) {
<div class="actions">
<button (click)="acceptSuggestion(suggestion)" class="btn btn-accept">
Accept
@@ -212,8 +206,8 @@ import { Suggestion, SuggestionStatus, SuggestionEntity, CreateGameDto, UpdateGa
}
@if (showDeclineModal()) {
<div class="modal-overlay" (click)="closeDeclineModal()" (keyup.escape)="closeDeclineModal()" tabindex="0" role="button">
<div class="modal" (click)="$event.stopPropagation()" (keyup)="$event.stopPropagation()" tabindex="-1">
<div class="modal-overlay" (click)="closeDeclineModal()">
<div class="modal" (click)="$event.stopPropagation()">
<h3>Decline Suggestion</h3>
<p>Are you sure you want to decline "{{ decliningsuggestion()?.title }}"?</p>
<div class="form-group">
@@ -235,64 +229,160 @@ import { Suggestion, SuggestionStatus, SuggestionEntity, CreateGameDto, UpdateGa
}
@if (showEditModal()) {
<div class="modal-overlay" (click)="closeEditModal()" (keyup.escape)="closeEditModal()" tabindex="0" role="button">
<div class="modal edit-modal" (click)="$event.stopPropagation()" (keyup)="$event.stopPropagation()" tabindex="-1">
<div class="modal-overlay" (click)="closeEditModal()">
<div class="modal edit-modal" (click)="$event.stopPropagation()">
<h3>Review & Edit Before Accepting</h3>
<p>Review and edit the details before adding to your collection.</p>
@if (editingSuggestion()) {
@if (editingSuggestion()!.entityType === SuggestionEntity.game) {
<h3>Review & Edit Game Before Accepting</h3>
<p>Review and edit all the details before adding to your collection.</p>
<app-game-form
mode="add"
[initialData]="getGameInitialData(editingSuggestion()!)"
(formSubmit)="saveGameFromSuggestion($event)"
(formCancel)="closeEditModal()"
></app-game-form>
} @else if (editingSuggestion()!.entityType === SuggestionEntity.book) {
<h3>Review & Edit Book Before Accepting</h3>
<p>Review and edit all the details before adding to your collection.</p>
<app-book-form
mode="add"
[initialData]="getBookInitialData(editingSuggestion()!)"
(formSubmit)="saveBookFromSuggestion($event)"
(formCancel)="closeEditModal()"
></app-book-form>
} @else if (editingSuggestion()!.entityType === SuggestionEntity.music) {
<h3>Review & Edit Music Before Accepting</h3>
<p>Review and edit all the details before adding to your collection.</p>
<app-music-form
mode="add"
[initialData]="getMusicInitialData(editingSuggestion()!)"
(formSubmit)="saveMusicFromSuggestion($event)"
(formCancel)="closeEditModal()"
></app-music-form>
} @else if (editingSuggestion()!.entityType === SuggestionEntity.show) {
<h3>Review & Edit Show Before Accepting</h3>
<p>Review and edit all the details before adding to your collection.</p>
<app-show-form
mode="add"
[initialData]="getShowInitialData(editingSuggestion()!)"
(formSubmit)="saveShowFromSuggestion($event)"
(formCancel)="closeEditModal()"
></app-show-form>
} @else if (editingSuggestion()!.entityType === SuggestionEntity.manga) {
<h3>Review & Edit Manga Before Accepting</h3>
<p>Review and edit all the details before adding to your collection.</p>
<app-manga-form
mode="add"
[initialData]="getMangaInitialData(editingSuggestion()!)"
(formSubmit)="saveMangaFromSuggestion($event)"
(formCancel)="closeEditModal()"
></app-manga-form>
} @else if (editingSuggestion()!.entityType === SuggestionEntity.art) {
<h3>Review & Edit Art Before Accepting</h3>
<p>Review and edit all the details before adding to your collection.</p>
<app-art-form
mode="add"
[initialData]="getArtInitialData(editingSuggestion()!)"
(formSubmit)="saveArtFromSuggestion($event)"
(formCancel)="closeEditModal()"
></app-art-form>
}
<form (ngSubmit)="confirmAcceptWithEdits()">
<div class="form-group">
<label for="edit-title">Title</label>
<input
type="text"
id="edit-title"
[(ngModel)]="editedData.title"
name="title"
required
>
</div>
@switch (editingSuggestion()!.entityType) {
@case ('BOOK') {
<div class="form-group">
<label for="edit-author">Author</label>
<input
type="text"
id="edit-author"
[(ngModel)]="editedData.author"
name="author"
required
>
</div>
<div class="form-group">
<label for="edit-isbn">ISBN</label>
<input
type="text"
id="edit-isbn"
[(ngModel)]="editedData.isbn"
name="isbn"
>
</div>
}
@case ('GAME') {
<div class="form-group">
<label for="edit-platform">Platform</label>
<input
type="text"
id="edit-platform"
[(ngModel)]="editedData.platform"
name="platform"
>
</div>
}
@case ('MUSIC') {
<div class="form-group">
<label for="edit-artist">Artist</label>
<input
type="text"
id="edit-artist"
[(ngModel)]="editedData.artist"
name="artist"
required
>
</div>
<div class="form-group">
<label for="edit-type">Type</label>
<select id="edit-type" [(ngModel)]="editedData.type" name="type" required>
<option value="ALBUM">Album</option>
<option value="SINGLE">Single</option>
<option value="EP">EP</option>
</select>
</div>
}
@case ('ART') {
<div class="form-group">
<label for="edit-artist">Artist</label>
<input
type="text"
id="edit-artist"
[(ngModel)]="editedData.artist"
name="artist"
required
>
</div>
<div class="form-group">
<label for="edit-description">Description</label>
<textarea
id="edit-description"
[(ngModel)]="editedData.description"
name="description"
rows="3"
></textarea>
</div>
<div class="form-group">
<label for="edit-imageUrl">Image URL</label>
<input
type="text"
id="edit-imageUrl"
[(ngModel)]="editedData.imageUrl"
name="imageUrl"
required
>
</div>
}
@case ('SHOW') {
<div class="form-group">
<label for="edit-type">Type</label>
<select id="edit-type" [(ngModel)]="editedData.type" name="type" required>
<option value="TV_SERIES">TV Series</option>
<option value="ANIME">Anime</option>
<option value="FILM">Film</option>
<option value="DOCUMENTARY">Documentary</option>
</select>
</div>
}
@case ('MANGA') {
<div class="form-group">
<label for="edit-author">Author</label>
<input
type="text"
id="edit-author"
[(ngModel)]="editedData.author"
name="author"
required
>
</div>
}
}
<div class="form-group">
<label for="edit-notes">Notes</label>
<textarea
id="edit-notes"
[(ngModel)]="editedData.notes"
name="notes"
rows="3"
></textarea>
</div>
@if (editingSuggestion()!.entityType !== 'ART') {
<div class="form-group">
<label for="edit-coverImage">Cover Image URL</label>
<input
type="text"
id="edit-coverImage"
[(ngModel)]="editedData.coverImage"
name="coverImage"
>
</div>
}
<div class="modal-actions">
<button type="submit" class="btn btn-accept">Accept with Edits</button>
<button type="button" (click)="closeEditModal()" class="btn btn-secondary">Cancel</button>
</div>
</form>
}
</div>
</div>
@@ -637,11 +727,10 @@ export class AdminSuggestionsComponent implements OnInit {
pageSize = signal(25);
SuggestionStatus = SuggestionStatus;
SuggestionEntity = SuggestionEntity;
unreviewedCount = () => this.suggestions().filter(s => s.status === SuggestionStatus.unreviewed).length;
acceptedCount = () => this.suggestions().filter(s => s.status === SuggestionStatus.accepted).length;
declinedCount = () => this.suggestions().filter(s => s.status === SuggestionStatus.declined).length;
unreviewedCount = () => this.suggestions().filter(s => s.status === SuggestionStatus.UNREVIEWED).length;
acceptedCount = () => this.suggestions().filter(s => s.status === SuggestionStatus.ACCEPTED).length;
declinedCount = () => this.suggestions().filter(s => s.status === SuggestionStatus.DECLINED).length;
filteredSuggestions = computed(() => {
const filter = this.statusFilter();
@@ -699,20 +788,20 @@ export class AdminSuggestionsComponent implements OnInit {
getStatusLabel(status: SuggestionStatus): string {
switch (status) {
case SuggestionStatus.unreviewed: return 'Pending';
case SuggestionStatus.accepted: return 'Accepted';
case SuggestionStatus.declined: return 'Declined';
case SuggestionStatus.UNREVIEWED: return 'Pending';
case SuggestionStatus.ACCEPTED: return 'Accepted';
case SuggestionStatus.DECLINED: return 'Declined';
}
}
getEntityIcon(entityType: SuggestionEntity): string {
switch (entityType) {
case SuggestionEntity.game: return '🎮';
case SuggestionEntity.book: return '📚';
case SuggestionEntity.music: return '🎵';
case SuggestionEntity.manga: return '📖';
case SuggestionEntity.show: return '📺';
case SuggestionEntity.art: return '🎨';
case SuggestionEntity.GAME: return '🎮';
case SuggestionEntity.BOOK: return '📚';
case SuggestionEntity.MUSIC: return '🎵';
case SuggestionEntity.MANGA: return '📖';
case SuggestionEntity.SHOW: return '📺';
case SuggestionEntity.ART: return '🎨';
}
}
@@ -720,174 +809,59 @@ export class AdminSuggestionsComponent implements OnInit {
return new Date(date).toLocaleDateString();
}
getGameInitialData(suggestion: Suggestion): Partial<CreateGameDto> {
const gameData = suggestion.gameData as any;
return {
title: suggestion.title,
platform: gameData?.platform || undefined,
status: GameStatus.backlog, // Default to backlog for new suggestions
notes: gameData?.notes || undefined,
coverImage: gameData?.coverImage || undefined,
tags: [],
links: []
};
}
async saveGameFromSuggestion(data: CreateGameDto | UpdateGameDto) {
const suggestion = this.editingSuggestion();
if (!suggestion) return;
try {
// Treat it as CreateGameDto since we're creating a new item from a suggestion
await this.suggestionService.acceptSuggestionWithEdits(suggestion.id, data as CreateGameDto);
alert(`"${(data as CreateGameDto).title}" has been added to your collection!`);
this.closeEditModal();
this.loadSuggestions();
} catch (error) {
alert('Failed to accept suggestion. Please try again.');
}
}
getBookInitialData(suggestion: Suggestion): Partial<CreateBookDto> {
const bookData = suggestion.bookData as any;
return {
title: suggestion.title,
author: bookData?.author || '',
isbn: bookData?.isbn || undefined,
status: BookStatus.toRead,
notes: bookData?.notes || undefined,
coverImage: bookData?.coverImage || undefined,
tags: [],
links: []
};
}
async saveBookFromSuggestion(data: CreateBookDto | UpdateBookDto) {
const suggestion = this.editingSuggestion();
if (!suggestion) return;
try {
await this.suggestionService.acceptSuggestionWithEdits(suggestion.id, data as CreateBookDto);
alert(`"${(data as CreateBookDto).title}" has been added to your collection!`);
this.closeEditModal();
this.loadSuggestions();
} catch (error) {
alert('Failed to accept suggestion. Please try again.');
}
}
getMusicInitialData(suggestion: Suggestion): Partial<CreateMusicDto> {
const musicData = suggestion.musicData as any;
return {
title: suggestion.title,
artist: musicData?.artist || '',
type: musicData?.type || MusicType.album,
status: MusicStatus.wantToListen,
notes: musicData?.notes || undefined,
coverArt: musicData?.coverArt || undefined,
tags: [],
links: []
};
}
async saveMusicFromSuggestion(data: CreateMusicDto | UpdateMusicDto) {
const suggestion = this.editingSuggestion();
if (!suggestion) return;
try {
await this.suggestionService.acceptSuggestionWithEdits(suggestion.id, data as CreateMusicDto);
alert(`"${(data as CreateMusicDto).title}" has been added to your collection!`);
this.closeEditModal();
this.loadSuggestions();
} catch (error) {
alert('Failed to accept suggestion. Please try again.');
}
}
getShowInitialData(suggestion: Suggestion): Partial<CreateShowDto> {
const showData = suggestion.showData as any;
return {
title: suggestion.title,
type: showData?.type || ShowType.tvSeries,
status: ShowStatus.wantToWatch,
notes: showData?.notes || undefined,
coverImage: showData?.coverImage || undefined,
tags: [],
links: []
};
}
async saveShowFromSuggestion(data: CreateShowDto | UpdateShowDto) {
const suggestion = this.editingSuggestion();
if (!suggestion) return;
try {
await this.suggestionService.acceptSuggestionWithEdits(suggestion.id, data as CreateShowDto);
alert(`"${(data as CreateShowDto).title}" has been added to your collection!`);
this.closeEditModal();
this.loadSuggestions();
} catch (error) {
alert('Failed to accept suggestion. Please try again.');
}
}
getMangaInitialData(suggestion: Suggestion): Partial<CreateMangaDto> {
const mangaData = suggestion.mangaData as any;
return {
title: suggestion.title,
author: mangaData?.author || '',
status: MangaStatus.wantToRead,
notes: mangaData?.notes || undefined,
coverImage: mangaData?.coverImage || undefined,
tags: [],
links: []
};
}
async saveMangaFromSuggestion(data: CreateMangaDto | UpdateMangaDto) {
const suggestion = this.editingSuggestion();
if (!suggestion) return;
try {
await this.suggestionService.acceptSuggestionWithEdits(suggestion.id, data as CreateMangaDto);
alert(`"${(data as CreateMangaDto).title}" has been added to your collection!`);
this.closeEditModal();
this.loadSuggestions();
} catch (error) {
alert('Failed to accept suggestion. Please try again.');
}
}
getArtInitialData(suggestion: Suggestion): Partial<CreateArtDto> {
const artData = suggestion.artData as any;
return {
title: suggestion.title,
artist: artData?.artist || '',
description: artData?.description || undefined,
imageUrl: artData?.imageUrl || '',
tags: [],
links: []
};
}
async saveArtFromSuggestion(data: CreateArtDto | UpdateArtDto) {
const suggestion = this.editingSuggestion();
if (!suggestion) return;
try {
await this.suggestionService.acceptSuggestionWithEdits(suggestion.id, data as CreateArtDto);
alert(`"${(data as CreateArtDto).title}" has been added to your collection!`);
this.closeEditModal();
this.loadSuggestions();
} catch (error) {
alert('Failed to accept suggestion. Please try again.');
}
}
async acceptSuggestion(suggestion: Suggestion) {
this.editingSuggestion.set(suggestion);
// For all entity types, we'll use the form components which have their own initialization logic
// Pre-populate the edit form with suggestion data
this.editedData = {
title: suggestion.title,
notes: '',
coverImage: '',
coverArt: ''
};
// Add entity-specific data
switch (suggestion.entityType) {
case 'BOOK':
const bookData = suggestion.bookData as any;
this.editedData.author = bookData?.author || '';
this.editedData.isbn = bookData?.isbn || '';
this.editedData.notes = bookData?.notes || '';
this.editedData.coverImage = bookData?.coverImage || '';
break;
case 'GAME':
const gameData = suggestion.gameData as any;
this.editedData.platform = gameData?.platform || '';
this.editedData.notes = gameData?.notes || '';
this.editedData.coverImage = gameData?.coverImage || '';
break;
case 'MUSIC':
const musicData = suggestion.musicData as any;
this.editedData.artist = musicData?.artist || '';
this.editedData.type = musicData?.type || 'ALBUM';
this.editedData.notes = musicData?.notes || '';
this.editedData.coverArt = musicData?.coverArt || '';
break;
case 'ART':
const artData = suggestion.artData as any;
this.editedData.artist = artData?.artist || '';
this.editedData.description = artData?.description || '';
this.editedData.imageUrl = artData?.imageUrl || '';
break;
case 'SHOW':
const showData = suggestion.showData as any;
this.editedData.type = showData?.type || 'TV_SERIES';
this.editedData.notes = showData?.notes || '';
this.editedData.coverImage = showData?.coverImage || '';
break;
case 'MANGA':
const mangaData = suggestion.mangaData as any;
this.editedData.author = mangaData?.author || '';
this.editedData.notes = mangaData?.notes || '';
this.editedData.coverImage = mangaData?.coverImage || '';
break;
}
this.showEditModal.set(true);
}
@@ -909,6 +883,20 @@ export class AdminSuggestionsComponent implements OnInit {
this.editedData = {};
}
async confirmAcceptWithEdits() {
const suggestion = this.editingSuggestion();
if (!suggestion) return;
try {
await this.suggestionService.acceptSuggestionWithEdits(suggestion.id, this.editedData);
alert(`"${this.editedData.title}" has been added to your collection with your edits!`);
this.closeEditModal();
this.loadSuggestions();
} catch (error) {
alert('Failed to accept suggestion. Please try again.');
}
}
async confirmDecline() {
const suggestion = this.decliningsuggestion();
if (!suggestion) return;
@@ -1,656 +0,0 @@
/**
* @copyright 2026 NHCarrigan
* @license Naomi's Public License
* @author Naomi Carrigan
*/
import { Component, OnInit, inject, signal } from '@angular/core';
import { CommonModule } from '@angular/common';
import { RouterLink, ActivatedRoute, Router } from '@angular/router';
import { FormsModule } from '@angular/forms';
import { ArtService } from '../../services/art.service';
import { CommentsService } from '../../services/comments.service';
import { AuthService } from '../../services/auth.service';
import { SanitizeService } from '../../services/sanitize.service';
import { CommentDisplayComponent } from '../comment-display/comment-display.component';
import { LikeButtonComponent } from '../shared/like-button.component';
import { ArtFormComponent } from '../shared/art-form.component';
import { Art, Comment, UpdateArtDto } from '@library/shared-types';
@Component({
selector: 'app-art-detail',
standalone: true,
imports: [CommonModule, RouterLink, FormsModule, CommentDisplayComponent, LikeButtonComponent, ArtFormComponent],
template: `
<div class="container">
<div class="breadcrumb">
<a routerLink="/art" class="breadcrumb-link">← Back to Art</a>
</div>
@if (showEditForm() && authService.user()?.isAdmin && art()) {
<app-art-form
mode="edit"
[art]="art()!"
(formSubmit)="saveEdit($event)"
(formCancel)="cancelEdit()"
></app-art-form>
}
@if (loading()) {
<div class="loading">Loading artwork details...</div>
} @else if (error()) {
<div class="error-state">
<h2>Artwork Not Found</h2>
<p>{{ error() }}</p>
<a routerLink="/art" class="btn btn-primary">Return to Art Gallery</a>
</div>
} @else if (art()) {
<div class="art-detail-card">
<div class="art-image-section">
<img [src]="art()!.imageUrl || '/assets/default-cover.jpg'" [alt]="art()!.description || art()!.title" class="art-image-large">
</div>
<div class="art-content">
<div class="art-header">
<h1>{{ art()!.title }}</h1>
</div>
<p class="artist">by {{ art()!.artist }}</p>
@if (authService.user()?.isAdmin) {
<div class="admin-actions">
<button (click)="toggleEditForm()" class="btn btn-edit">✏️ {{ showEditForm() ? 'Cancel Edit' : 'Edit' }}</button>
<button (click)="deleteArt()" class="btn btn-delete">🗑️ Delete</button>
</div>
}
<div class="info-row">
<span class="info-label">Added:</span>
<span class="info-value">{{ formatDate(art()!.createdAt) }}</span>
</div>
<div class="info-row">
<span class="info-label">Last Updated:</span>
<span class="info-value">{{ formatDate(art()!.updatedAt) }}</span>
</div>
<div class="like-section">
<app-like-button
entityType="art"
[entityId]="art()!.id"
></app-like-button>
</div>
@if (art()!.description) {
<div class="notes-section">
<h3>Description</h3>
<p class="notes">{{ art()!.description }}</p>
</div>
}
@if (art()!.tags && art()!.tags.length > 0) {
<div class="tags-section">
<h3>Tags</h3>
<div class="tags-display">
@for (tag of art()!.tags; track tag) {
<span class="tag-chip">{{ tag }}</span>
}
</div>
</div>
}
@if (art()!.links && art()!.links.length > 0) {
<div class="links-section">
<h3>External Links</h3>
<div class="links-display">
@for (link of art()!.links; track link.url) {
<a [href]="link.url" target="_blank" rel="noopener noreferrer" class="external-link">
{{ link.title }} ↗
</a>
}
</div>
</div>
}
<div class="comments-section">
<h3>Comments</h3>
@if (authService.isAuthenticated()) {
@if (authService.user()?.isBanned) {
<div class="banned-notice">
You have been banned from commenting.
</div>
} @else {
<form (ngSubmit)="addComment()" class="comment-form">
<textarea
[(ngModel)]="newCommentContent"
name="comment"
placeholder="Add a comment (Markdown supported)..."
rows="3"
></textarea>
<button type="submit" class="btn btn-primary">Post Comment</button>
</form>
}
} @else {
<div class="auth-prompt">
<p>Please sign in to comment.</p>
</div>
}
@if (commentsLoading()) {
<div class="comments-loading">Loading comments...</div>
} @else {
<app-comment-display
[comments]="comments"
(edit)="handleCommentEdit($event)"
(delete)="deleteComment($event)"
/>
}
</div>
</div>
</div>
}
</div>
`,
styles: [`
.container {
max-width: 900px;
margin: 0 auto;
padding: 2rem;
}
.breadcrumb {
margin-bottom: 1.5rem;
}
.breadcrumb-link {
color: #ec4899;
text-decoration: none;
font-weight: 500;
transition: opacity 0.3s;
}
.breadcrumb-link:hover {
opacity: 0.8;
}
.loading {
text-align: center;
padding: 3rem;
color: #666;
font-size: 1.1rem;
}
.error-state {
text-align: center;
padding: 3rem;
background: #fef2f2;
border: 1px solid #fecaca;
border-radius: 8px;
}
.error-state h2 {
color: #991b1b;
margin-bottom: 1rem;
}
.error-state p {
color: #dc2626;
margin-bottom: 1.5rem;
}
.art-detail-card {
background: white;
border: 1px solid #e5e7eb;
border-radius: 8px;
overflow: hidden;
box-shadow: 0 1px 3px rgba(0,0,0,0.1);
}
.art-image-section {
width: 100%;
background: #f3f4f6;
display: flex;
justify-content: center;
align-items: center;
padding: 2rem;
}
.art-image-large {
max-width: 100%;
max-height: 600px;
object-fit: contain;
border-radius: 4px;
}
.art-content {
padding: 2rem;
}
.art-header {
display: flex;
align-items: center;
gap: 1rem;
margin-bottom: 1rem;
flex-wrap: wrap;
}
.art-header h1 {
margin: 0;
font-size: 2rem;
color: #1f2937;
}
.artist {
color: #6b7280;
font-weight: 500;
font-size: 1.1rem;
margin: 0 0 1rem 0;
}
.admin-actions {
display: flex;
gap: 0.75rem;
margin-bottom: 1.5rem;
padding: 1rem;
background: #fef3c7;
border-radius: 4px;
border: 1px solid #fbbf24;
}
.btn-edit {
background: #3b82f6;
color: white;
}
.btn-edit:hover {
background: #2563eb;
}
.btn-delete {
background: #ef4444;
color: white;
}
.btn-delete:hover {
background: #dc2626;
}
.info-row {
display: flex;
align-items: center;
gap: 1rem;
margin-bottom: 0.75rem;
padding: 0.5rem 0;
border-bottom: 1px solid #f3f4f6;
}
.info-label {
font-weight: 600;
color: #4b5563;
min-width: 120px;
}
.info-value {
color: #1f2937;
}
.rating {
display: flex;
align-items: center;
gap: 0.25rem;
}
.rating span {
color: #e5e7eb;
font-size: 1.2rem;
}
.rating span.filled {
color: #f59e0b;
}
.rating-text {
margin-left: 0.5rem;
font-size: 1rem;
color: #6b7280;
font-weight: 500;
}
.like-section {
margin: 1.5rem 0;
}
.notes-section,
.tags-section,
.links-section {
margin-top: 2rem;
}
.notes-section h3,
.tags-section h3,
.links-section h3,
.comments-section h3 {
font-size: 1.25rem;
color: #1f2937;
margin-bottom: 1rem;
}
.notes {
font-size: 1rem;
color: #4b5563;
line-height: 1.6;
white-space: pre-wrap;
}
.tags-display {
display: flex;
flex-wrap: wrap;
gap: 0.5rem;
}
.tag-chip {
background: #ec4899;
color: white;
padding: 0.375rem 0.875rem;
border-radius: 12px;
font-size: 0.875rem;
font-weight: 500;
}
.links-display {
display: flex;
flex-wrap: wrap;
gap: 0.75rem;
}
.external-link {
color: #ec4899;
text-decoration: none;
font-size: 0.95rem;
padding: 0.5rem 1rem;
border: 2px solid #ec4899;
border-radius: 4px;
transition: all 0.2s;
font-weight: 500;
}
.external-link:hover {
background: #ec4899;
color: white;
}
.comments-section {
margin-top: 2rem;
padding-top: 2rem;
border-top: 2px solid #e5e7eb;
}
.comment-form {
margin-bottom: 1.5rem;
}
.comment-form textarea {
width: 100%;
padding: 0.75rem;
border: 1px solid #d1d5db;
border-radius: 4px;
font-size: 1rem;
resize: vertical;
margin-bottom: 0.75rem;
font-family: inherit;
}
.comment-form textarea:focus {
outline: none;
border-color: #ec4899;
}
.banned-notice {
background: #fef2f2;
border: 1px solid #fecaca;
color: #991b1b;
padding: 1rem;
border-radius: 4px;
text-align: center;
margin-bottom: 1.5rem;
font-size: 0.95rem;
}
.auth-prompt {
background: #f3f4f6;
padding: 1rem;
border-radius: 4px;
text-align: center;
margin-bottom: 1.5rem;
}
.auth-prompt p {
margin: 0;
color: #4b5563;
}
.comments-loading {
text-align: center;
padding: 1.5rem;
color: #6b7280;
font-size: 0.95rem;
}
.btn {
padding: 0.625rem 1.25rem;
border: none;
border-radius: 4px;
cursor: pointer;
font-size: 1rem;
transition: all 0.3s;
font-weight: 500;
}
.btn:hover {
opacity: 0.9;
}
.btn-primary {
background: #ec4899;
color: white;
}
@media (max-width: 640px) {
.container {
padding: 1rem;
}
.art-header {
flex-direction: column;
align-items: flex-start;
}
.art-header h1 {
font-size: 1.5rem;
}
.admin-actions {
display: flex;
gap: 0.75rem;
margin-bottom: 1.5rem;
padding: 1rem;
background: #fef3c7;
border-radius: 4px;
border: 1px solid #fbbf24;
}
.btn-edit {
background: #3b82f6;
color: white;
}
.btn-edit:hover {
background: #2563eb;
}
.btn-delete {
background: #ef4444;
color: white;
}
.btn-delete:hover {
background: #dc2626;
}
.info-row {
flex-direction: column;
align-items: flex-start;
gap: 0.25rem;
}
.info-label {
min-width: auto;
}
}
`]
})
export class ArtDetailComponent implements OnInit {
private readonly artService = inject(ArtService);
private readonly commentsService = inject(CommentsService);
readonly authService = inject(AuthService);
private readonly sanitizeService = inject(SanitizeService);
private readonly route = inject(ActivatedRoute);
private readonly router = inject(Router);
art = signal<Art | null>(null);
comments = signal<Comment[]>([]);
loading = signal(true);
commentsLoading = signal(false);
error = signal<string | null>(null);
newCommentContent = '';
showEditForm = signal(false);
ngOnInit() {
const artId = this.route.snapshot.paramMap.get('id');
if (!artId) {
this.error.set('No artwork ID provided');
this.loading.set(false);
return;
}
this.loadArt(artId);
this.loadComments(artId);
}
private loadArt(artId: string) {
this.loading.set(true);
this.artService.getArtById(artId).subscribe({
next: (art) => {
if (!art) {
this.error.set('Artwork not found');
} else {
this.art.set(art);
}
this.loading.set(false);
},
error: () => {
this.error.set('Failed to load artwork. It may not exist or there was an error.');
this.loading.set(false);
}
});
}
private loadComments(artId: string) {
this.commentsLoading.set(true);
this.commentsService.getCommentsForArt(artId).subscribe({
next: (comments) => {
this.comments.set(comments);
this.commentsLoading.set(false);
},
error: () => {
this.commentsLoading.set(false);
}
});
}
addComment() {
const art = this.art();
if (!art || !this.newCommentContent.trim()) return;
this.commentsService.addCommentToArt(art.id, { content: this.newCommentContent }).subscribe({
next: (comment) => {
this.comments.set([comment, ...this.comments()]);
this.newCommentContent = '';
}
});
}
handleCommentEdit(event: { commentId: string; content: string }) {
const art = this.art();
if (!art) return;
this.commentsService.updateCommentOnArt(art.id, event.commentId, event.content).subscribe({
next: (updatedComment) => {
this.comments.set(
this.comments().map(c => c.id === event.commentId ? updatedComment : c)
);
}
});
}
deleteComment(commentId: string) {
const art = this.art();
if (!art || !confirm('Are you sure you want to delete this comment?')) return;
this.commentsService.deleteCommentFromArt(art.id, commentId).subscribe({
next: () => {
this.comments.set(this.comments().filter(c => c.id !== commentId));
}
});
}
formatDate(date: Date | string): string {
return new Date(date).toLocaleDateString('en-GB', {
year: 'numeric',
month: 'long',
day: 'numeric'
});
}
toggleEditForm() {
this.showEditForm.update(value => !value);
}
saveEdit(data: UpdateArtDto) {
const art = this.art();
if (!art) return;
this.artService.updateArt(art.id, data).subscribe({
next: (updatedArt) => {
this.art.set(updatedArt);
this.showEditForm.set(false);
},
error: (err) => {
alert('Failed to update art: ' + (err.error?.message || 'Unknown error'));
}
});
}
cancelEdit() {
this.showEditForm.set(false);
}
deleteArt() {
const art = this.art();
if (!art) return;
if (!confirm(`Are you sure you want to delete "${art.title}"? This action cannot be undone.`)) {
return;
}
this.artService.deleteArt(art.id).subscribe({
next: () => {
this.router.navigate(['/art']);
},
error: (err) => {
alert('Failed to delete art: ' + (err.error?.message || 'Unknown error'));
}
});
}
}
@@ -7,7 +7,6 @@
import { Component, OnInit, inject, signal, computed } from '@angular/core';
import { CommonModule } from '@angular/common';
import { FormsModule } from '@angular/forms';
import { RouterModule } from '@angular/router';
import { ArtService } from '../../services/art.service';
import { AuthService } from '../../services/auth.service';
import { CommentsService } from '../../services/comments.service';
@@ -20,13 +19,9 @@ import { Art, CreateArtDto, UpdateArtDto, Comment, SuggestionEntity, Link } from
@Component({
selector: 'app-art-gallery',
standalone: true,
imports: [CommonModule, FormsModule, RouterModule, PaginationComponent, LikeButtonComponent],
imports: [CommonModule, FormsModule, PaginationComponent, LikeButtonComponent],
template: `
<div class="container">
<div class="page-hero">
<img src="/assets/avatars/art-avatar.jpg" alt="Art avatar" class="page-avatar" />
</div>
<div class="header-section">
<h2>Art Gallery</h2>
<p class="subtitle">Artwork of Naomi</p>
@@ -110,7 +105,8 @@ import { Art, CreateArtDto, UpdateArtDto, Comment, SuggestionEntity, Link } from
}
<div class="form-group">
<div class="tags-input-container" aria-label="Tags">
<label>Tags</label>
<div class="tags-input-container">
@for (tag of newArt.tags; track tag; let i = $index) {
<span class="tag">
{{ tag }}
@@ -127,7 +123,8 @@ import { Art, CreateArtDto, UpdateArtDto, Comment, SuggestionEntity, Link } from
</div>
</div>
<div class="form-group" aria-label="External Links">
<div class="form-group">
<label>External Links</label>
<div class="links-list">
@for (link of newArt.links; track link.url; let i = $index) {
<div class="link-item">
@@ -283,7 +280,8 @@ import { Art, CreateArtDto, UpdateArtDto, Comment, SuggestionEntity, Link } from
}
<div class="form-group">
<div class="tags-input-container" aria-label="Tags">
<label>Tags</label>
<div class="tags-input-container">
@for (tag of editArt.tags; track tag; let i = $index) {
<span class="tag">
{{ tag }}
@@ -300,7 +298,8 @@ import { Art, CreateArtDto, UpdateArtDto, Comment, SuggestionEntity, Link } from
</div>
</div>
<div class="form-group" aria-label="External Links">
<div class="form-group">
<label>External Links</label>
<div class="links-list">
@for (link of editArt.links; track link.url; let i = $index) {
<div class="link-item">
@@ -396,29 +395,45 @@ import { Art, CreateArtDto, UpdateArtDto, Comment, SuggestionEntity, Link } from
<div class="gallery-grid">
@for (art of paginatedArtPieces(); track art.id) {
<div class="art-card">
<a [routerLink]="['/art', art.id]" class="card-link">
<div class="art-image-container">
<img
[src]="art.imageUrl || '/assets/default-cover.jpg'"
[alt]="art.description || art.title"
class="art-image"
>
</div>
<div class="art-image-container">
<img
[src]="art.imageUrl"
[alt]="art.description || art.title"
class="art-image"
(click)="openLightbox(art)"
>
</div>
<div class="art-info">
<h3>{{ art.title }}</h3>
<p class="artist">by {{ art.artist }}</p>
</div>
</a>
<div class="art-info">
<h3>{{ art.title }}</h3>
<p class="artist">by {{ art.artist }}</p>
<p class="date-added">Added: {{ formatDate(art.dateAdded) }}</p>
<div class="card-actions">
<app-like-button
entityType="art"
[entityId]="art.id"
></app-like-button>
@if (art.tags && art.tags.length > 0) {
<div class="tags-display">
@for (tag of art.tags; track tag) {
<span class="tag-chip">{{ tag }}</span>
}
</div>
}
@if (art.links && art.links.length > 0) {
<div class="links-display">
@for (link of art.links; track link.url) {
<a [href]="link.url" target="_blank" rel="noopener noreferrer" class="external-link">
{{ link.title }} ↗
</a>
}
</div>
}
@if (authService.isAdmin()) {
<div class="admin-actions">
<div class="actions">
<button (click)="startEdit(art)" class="btn btn-secondary btn-sm">
Edit
</button>
@@ -427,6 +442,85 @@ import { Art, CreateArtDto, UpdateArtDto, Comment, SuggestionEntity, Link } from
</button>
</div>
}
<div class="comments-section">
<button (click)="toggleComments(art.id)" class="btn btn-secondary btn-sm comments-toggle">
{{ expandedComments()[art.id] ? 'Hide' : 'Show' }} Comments{{ comments()[art.id] ? ' (' + getCommentCount(art.id) + ')' : '' }}
</button>
@if (expandedComments()[art.id]) {
<div class="comments-container">
@if (authService.isAuthenticated()) {
@if (authService.user()?.isBanned) {
<div class="banned-notice">
You have been banned from commenting.
</div>
} @else {
<form (ngSubmit)="addComment(art.id)" class="comment-form">
<textarea
[(ngModel)]="newCommentContent[art.id]"
name="comment"
placeholder="Add a comment (Markdown supported)..."
rows="2"
></textarea>
<button type="submit" class="btn btn-primary btn-sm">Post Comment</button>
</form>
}
}
@if (commentsLoading()[art.id]) {
<div class="comments-loading">Loading comments...</div>
} @else {
@for (comment of comments()[art.id] || []; track comment.id) {
<div class="comment">
<div class="comment-header">
@if (comment.user.avatar) {
<img [src]="comment.user.avatar" [alt]="comment.user.username" class="comment-avatar">
}
<span class="comment-author">{{ comment.user.username }}</span>
@if (comment.user.inDiscord) {
<span class="discord-badge">Discord</span>
}
@if (comment.user.isVip) {
<span class="vip-badge">VIP</span>
}
@if (comment.user.isMod) {
<span class="mod-badge">Mod</span>
}
@if (comment.user.isStaff) {
<span class="staff-badge">Staff</span>
}
<span class="comment-date">{{ formatDate(comment.createdAt) }}</span>
@if (canEditComment(comment)) {
<button (click)="startEditComment(art.id, comment)" class="btn btn-secondary btn-xs">Edit</button>
}
@if (canDeleteComment(comment)) {
<button (click)="deleteComment(art.id, comment.id)" class="btn btn-danger btn-xs">Delete</button>
}
</div>
@if (editingCommentId() === comment.id) {
<div class="comment-edit-form">
<textarea
[(ngModel)]="editCommentContent"
name="editComment"
rows="3"
></textarea>
<div class="comment-edit-actions">
<button (click)="saveCommentEdit(art.id, comment.id)" class="btn btn-primary btn-xs">Save</button>
<button (click)="cancelCommentEdit()" class="btn btn-secondary btn-xs">Cancel</button>
</div>
</div>
} @else {
<div class="comment-content" [innerHTML]="sanitizeService.sanitizeHtml(comment.content)"></div>
}
</div>
} @empty {
<div class="no-comments">No comments yet. Be the first to comment!</div>
}
}
</div>
}
</div>
</div>
</div>
}
@@ -442,8 +536,8 @@ import { Art, CreateArtDto, UpdateArtDto, Comment, SuggestionEntity, Link } from
}
@if (lightboxArt()) {
<div class="lightbox" (click)="closeLightbox()" (keyup.escape)="closeLightbox()" tabindex="0" role="button">
<div class="lightbox-content" (click)="$event.stopPropagation()" (keyup)="$event.stopPropagation()" tabindex="-1">
<div class="lightbox" (click)="closeLightbox()">
<div class="lightbox-content" (click)="$event.stopPropagation()">
<button class="lightbox-close" (click)="closeLightbox()">&times;</button>
<img [src]="lightboxArt()!.imageUrl" [alt]="lightboxArt()!.description || lightboxArt()!.title">
<div class="lightbox-info">
@@ -462,26 +556,6 @@ import { Art, CreateArtDto, UpdateArtDto, Comment, SuggestionEntity, Link } from
padding: 2rem;
}
.page-hero {
text-align: center;
margin-bottom: 2rem;
}
.page-avatar {
width: 120px;
height: 120px;
border-radius: 50%;
object-fit: cover;
border: 3px solid #fdcb6e;
box-shadow: 0 4px 12px rgba(253, 203, 110, 0.3);
transition: all 0.3s;
}
.page-avatar:hover {
transform: scale(1.05);
box-shadow: 0 8px 16px rgba(253, 203, 110, 0.5);
}
.header-section {
display: flex;
flex-direction: column;
@@ -695,8 +769,6 @@ import { Art, CreateArtDto, UpdateArtDto, Comment, SuggestionEntity, Link } from
box-shadow: 0 4px 12px var(--witch-shadow);
border: 2px solid var(--witch-lavender);
transition: transform 0.3s, box-shadow 0.3s;
display: flex;
flex-direction: column;
}
.art-card:hover {
@@ -704,17 +776,11 @@ import { Art, CreateArtDto, UpdateArtDto, Comment, SuggestionEntity, Link } from
box-shadow: 0 8px 24px var(--witch-shadow);
}
.card-link {
text-decoration: none;
color: inherit;
display: block;
flex: 1;
}
.art-image-container {
width: 100%;
aspect-ratio: 1;
overflow: hidden;
cursor: pointer;
}
.art-image {
@@ -724,7 +790,7 @@ import { Art, CreateArtDto, UpdateArtDto, Comment, SuggestionEntity, Link } from
transition: transform 0.3s;
}
.card-link:hover .art-image {
.art-image:hover {
transform: scale(1.05);
}
@@ -740,19 +806,19 @@ import { Art, CreateArtDto, UpdateArtDto, Comment, SuggestionEntity, Link } from
.artist {
color: var(--witch-mauve);
font-style: italic;
margin: 0;
margin: 0 0 0.5rem 0;
}
.card-actions {
padding: 0 1rem 1rem 1rem;
display: flex;
flex-direction: column;
gap: 0.75rem;
.date-added {
font-size: 0.85rem;
color: var(--witch-mauve);
margin: 0 0 1rem 0;
}
.admin-actions {
.actions {
display: flex;
gap: 0.5rem;
margin-bottom: 1rem;
}
.btn {
@@ -847,6 +913,159 @@ import { Art, CreateArtDto, UpdateArtDto, Comment, SuggestionEntity, Link } from
opacity: 0.8;
}
/* Comments */
.comments-section {
margin-top: 1rem;
border-top: 1px solid var(--witch-lavender);
padding-top: 1rem;
}
.comments-toggle {
width: 100%;
}
.comments-container {
margin-top: 1rem;
}
.comment-form {
margin-bottom: 1rem;
}
.comment-form textarea {
width: 100%;
padding: 0.5rem;
border: 2px solid var(--witch-lavender);
border-radius: 4px;
font-size: 0.9rem;
background-color: var(--witch-moon);
color: var(--witch-purple);
resize: vertical;
margin-bottom: 0.5rem;
}
.comment-form textarea:focus {
outline: none;
border-color: var(--witch-rose);
}
.comment {
background: var(--witch-moon);
border: 1px solid var(--witch-lavender);
border-radius: 4px;
padding: 0.75rem;
margin-bottom: 0.5rem;
}
.comment-header {
display: flex;
align-items: center;
gap: 0.5rem;
margin-bottom: 0.5rem;
flex-wrap: wrap;
}
.comment-avatar {
width: 24px;
height: 24px;
border-radius: 50%;
}
.comment-author {
font-weight: 500;
color: var(--witch-plum);
}
.discord-badge {
background: #5865f2;
color: white;
padding: 0.1rem 0.4rem;
border-radius: 3px;
font-size: 0.7rem;
font-weight: 500;
}
.vip-badge {
background: linear-gradient(135deg, #ffd700, #ffaa00);
color: #1a1a1a;
padding: 0.1rem 0.4rem;
border-radius: 3px;
font-size: 0.7rem;
font-weight: 600;
}
.mod-badge {
background: linear-gradient(135deg, #00b894, #00cec9);
color: white;
padding: 0.1rem 0.4rem;
border-radius: 3px;
font-size: 0.7rem;
font-weight: 600;
}
.staff-badge {
background: linear-gradient(135deg, #e84393, #fd79a8);
color: white;
padding: 0.1rem 0.4rem;
border-radius: 3px;
font-size: 0.7rem;
font-weight: 600;
}
.comment-date {
font-size: 0.75rem;
color: var(--witch-mauve);
}
.comment-content {
font-size: 0.9rem;
color: var(--witch-purple);
}
.comments-loading,
.no-comments {
text-align: center;
padding: 1rem;
color: var(--witch-mauve);
font-size: 0.9rem;
}
.banned-notice {
background: #fef2f2;
border: 1px solid #fecaca;
color: #991b1b;
padding: 0.75rem 1rem;
border-radius: 4px;
text-align: center;
margin-bottom: 1rem;
font-size: 0.9rem;
}
.comment-edit-form {
margin-top: 0.5rem;
}
.comment-edit-form textarea {
width: 100%;
padding: 0.5rem;
border: 2px solid var(--witch-lavender);
border-radius: 4px;
font-size: 0.9rem;
background-color: var(--witch-moon);
color: var(--witch-purple);
resize: vertical;
}
.comment-edit-form textarea:focus {
outline: none;
border-color: var(--witch-rose);
}
.comment-edit-actions {
display: flex;
gap: 0.5rem;
margin-top: 0.5rem;
}
.tags-input-container {
display: flex;
@@ -896,6 +1115,21 @@ import { Art, CreateArtDto, UpdateArtDto, Comment, SuggestionEntity, Link } from
opacity: 0.8;
}
.tags-display {
display: flex;
flex-wrap: wrap;
gap: 0.5rem;
margin: 0.5rem 0;
}
.tag-chip {
background: rgba(253, 203, 110, 0.2);
color: #b38f00;
padding: 0.2rem 0.5rem;
border-radius: 4px;
font-size: 0.8rem;
}
.links-list {
margin-bottom: 0.5rem;
}
@@ -920,6 +1154,28 @@ import { Art, CreateArtDto, UpdateArtDto, Comment, SuggestionEntity, Link } from
flex: 1;
min-width: 120px;
}
.links-display {
display: flex;
flex-wrap: wrap;
gap: 0.5rem;
margin: 0.5rem 0;
}
.external-link {
color: #b38f00;
text-decoration: none;
font-size: 0.85rem;
padding: 0.2rem 0.5rem;
background: rgba(253, 203, 110, 0.1);
border-radius: 4px;
transition: background 0.2s;
}
.external-link:hover {
background: rgba(253, 203, 110, 0.2);
text-decoration: underline;
}
`]
})
export class ArtGalleryComponent implements OnInit {
@@ -1145,9 +1401,6 @@ export class ArtGalleryComponent implements OnInit {
this.editTagInput = '';
this.editLinkTitle = '';
this.editLinkUrl = '';
// Scroll to top so the form is visible
window.scrollTo({ top: 0, behavior: 'smooth' });
}
cancelEdit() {
@@ -1318,7 +1571,7 @@ export class ArtGalleryComponent implements OnInit {
try {
await this.suggestionService.createSuggestion({
entityType: SuggestionEntity.art,
entityType: SuggestionEntity.ART,
title: this.suggestedArt.title,
artist: this.suggestedArt.artist,
imageUrl: this.suggestedArt.imageUrl,
@@ -1362,21 +1615,4 @@ export class ArtGalleryComponent implements OnInit {
toggleFilters() {
this.showFilters.update(v => !v);
}
handleCommentEdit(artId: string, event: { commentId: string; content: string }) {
this.commentsService.updateCommentOnArt(artId, event.commentId, event.content).subscribe({
next: (updatedComment) => {
this.comments.set({
...this.comments(),
[artId]: (this.comments()[artId] || []).map(c =>
c.id === event.commentId ? updatedComment : c
)
});
}
});
}
getCommentsSignal(artId: string) {
return signal(this.comments()[artId] || []);
}
}
@@ -1,739 +0,0 @@
/**
* @copyright 2026 NHCarrigan
* @license Naomi's Public License
* @author Naomi Carrigan
*/
import { Component, OnInit, inject, signal } from '@angular/core';
import { CommonModule } from '@angular/common';
import { RouterLink, ActivatedRoute, Router } from '@angular/router';
import { FormsModule } from '@angular/forms';
import { BooksService } from '../../services/books.service';
import { CommentsService } from '../../services/comments.service';
import { AuthService } from '../../services/auth.service';
import { SanitizeService } from '../../services/sanitize.service';
import { CommentDisplayComponent } from '../comment-display/comment-display.component';
import { LikeButtonComponent } from '../shared/like-button.component';
import { BookFormComponent } from '../shared/book-form.component';
import { Book, Comment, BookStatus, UpdateBookDto } from '@library/shared-types';
@Component({
selector: 'app-book-detail',
standalone: true,
imports: [CommonModule, RouterLink, FormsModule, CommentDisplayComponent, LikeButtonComponent, BookFormComponent],
template: `
<div class="container">
<div class="breadcrumb">
<a routerLink="/books" class="breadcrumb-link">← Back to Books</a>
</div>
@if (showEditForm() && authService.user()?.isAdmin && book()) {
<app-book-form
mode="edit"
[book]="book()!"
(formSubmit)="saveEdit($event)"
(formCancel)="cancelEdit()"
></app-book-form>
}
@if (loading()) {
<div class="loading">Loading book details...</div>
} @else if (error()) {
<div class="error-state">
<h2>Book Not Found</h2>
<p>{{ error() }}</p>
<a routerLink="/books" class="btn btn-primary">Return to Books</a>
</div>
} @else if (book()) {
<div class="book-detail-card">
<div class="book-cover-section">
<img [src]="book()!.coverImage || '/assets/default-cover.jpg'" [alt]="book()!.title" class="book-cover-large">
</div>
<div class="book-content">
<div class="book-header">
<h1>{{ book()!.title }}</h1>
<span class="status status-{{ book()!.status }}">
{{ getStatusLabel(book()!.status) }}
</span>
</div>
<p class="author">by {{ book()!.author }}</p>
@if (authService.user()?.isAdmin) {
<div class="admin-actions">
<button (click)="editBook()" class="btn btn-edit">✏️ Edit</button>
<button (click)="deleteBook()" class="btn btn-delete">🗑️ Delete</button>
</div>
}
@if (book()!.series) {
<p class="series">
📚 {{ book()!.series }}@if (book()!.seriesOrder) { #{{ book()!.seriesOrder }}}
</p>
}
@if (book()!.isbn) {
<div class="info-row">
<span class="info-label">ISBN:</span>
<span class="info-value">{{ book()!.isbn }}</span>
</div>
}
@if (book()!.rating) {
<div class="info-row">
<span class="info-label">Rating:</span>
<div class="rating">
@for (star of [1,2,3,4,5,6,7,8,9,10]; track star) {
<span [class.filled]="star <= book()!.rating!">★</span>
}
<span class="rating-text">({{ book()!.rating }}/10)</span>
</div>
</div>
}
@if (book()!.timeSpent) {
<div class="info-row">
<span class="info-label">Reading Time:</span>
<span class="info-value time-spent">{{ formatTimeSpent(book()!.timeSpent!) }}</span>
</div>
}
@if (book()!.dateStarted) {
<div class="info-row">
<span class="info-label">Started:</span>
<span class="info-value">{{ formatDate(book()!.dateStarted!) }}</span>
</div>
}
@if (book()!.dateFinished) {
<div class="info-row">
<span class="info-label">Finished:</span>
<span class="info-value">{{ formatDate(book()!.dateFinished!) }}</span>
</div>
}
<div class="info-row">
<span class="info-label">Added:</span>
<span class="info-value">{{ formatDate(book()!.createdAt) }}</span>
</div>
<div class="info-row">
<span class="info-label">Last Updated:</span>
<span class="info-value">{{ formatDate(book()!.updatedAt) }}</span>
</div>
<div class="like-section">
<app-like-button
entityType="book"
[entityId]="book()!.id"
></app-like-button>
</div>
@if (book()!.notes) {
<div class="notes-section">
<h3>Notes</h3>
<p class="notes">{{ book()!.notes }}</p>
</div>
}
@if (book()!.tags && book()!.tags.length > 0) {
<div class="tags-section">
<h3>Tags</h3>
<div class="tags-display">
@for (tag of book()!.tags; track tag) {
<span class="tag-chip">{{ tag }}</span>
}
</div>
</div>
}
@if (book()!.links && book()!.links.length > 0) {
<div class="links-section">
<h3>External Links</h3>
<div class="links-display">
@for (link of book()!.links; track link.url) {
<a [href]="link.url" target="_blank" rel="noopener noreferrer" class="external-link">
{{ link.title }} ↗
</a>
}
</div>
</div>
}
<div class="comments-section">
<h3>Comments</h3>
@if (authService.isAuthenticated()) {
@if (authService.user()?.isBanned) {
<div class="banned-notice">
You have been banned from commenting.
</div>
} @else {
<form (ngSubmit)="addComment()" class="comment-form">
<textarea
[(ngModel)]="newCommentContent"
name="comment"
placeholder="Add a comment (Markdown supported)..."
rows="3"
></textarea>
<button type="submit" class="btn btn-primary">Post Comment</button>
</form>
}
} @else {
<div class="auth-prompt">
<p>Please sign in to comment.</p>
</div>
}
@if (commentsLoading()) {
<div class="comments-loading">Loading comments...</div>
} @else {
<app-comment-display
[comments]="comments"
(edit)="handleCommentEdit($event)"
(delete)="deleteComment($event)"
/>
}
</div>
</div>
</div>
}
</div>
`,
styles: [`
.container {
max-width: 900px;
margin: 0 auto;
padding: 2rem;
}
.breadcrumb {
margin-bottom: 1.5rem;
}
.breadcrumb-link {
color: #8b6f47;
text-decoration: none;
font-weight: 500;
transition: opacity 0.3s;
}
.breadcrumb-link:hover {
opacity: 0.8;
}
.loading {
text-align: center;
padding: 3rem;
color: #666;
font-size: 1.1rem;
}
.error-state {
text-align: center;
padding: 3rem;
background: #fef2f2;
border: 1px solid #fecaca;
border-radius: 8px;
}
.error-state h2 {
color: #991b1b;
margin-bottom: 1rem;
}
.error-state p {
color: #dc2626;
margin-bottom: 1.5rem;
}
.book-detail-card {
background: white;
border: 1px solid #e5e7eb;
border-radius: 8px;
overflow: hidden;
box-shadow: 0 1px 3px rgba(0,0,0,0.1);
}
.book-cover-section {
width: 100%;
background: #f3f4f6;
display: flex;
justify-content: center;
align-items: center;
padding: 2rem;
}
.book-cover-large {
max-width: 100%;
max-height: 400px;
object-fit: contain;
border-radius: 4px;
}
.book-content {
padding: 2rem;
}
.book-header {
display: flex;
align-items: center;
gap: 1rem;
margin-bottom: 1rem;
flex-wrap: wrap;
}
.book-header h1 {
margin: 0;
font-size: 2rem;
color: #1f2937;
}
.author {
color: #6b7280;
font-style: italic;
font-size: 1.1rem;
margin: 0 0 1rem 0;
}
.status {
display: inline-block;
padding: 0.5rem 1rem;
border-radius: 4px;
font-size: 0.9rem;
font-weight: 500;
}
.status-reading {
background: #fef3c7;
color: #92400e;
}
.status-finished {
background: #d1fae5;
color: #065f46;
}
.status-toRead {
background: #e0e7ff;
color: #3730a3;
}
.status-retired {
background: #f3f4f6;
color: #4b5563;
}
.admin-actions {
display: flex;
gap: 0.75rem;
margin-bottom: 1.5rem;
padding: 1rem;
background: #fef3c7;
border-radius: 4px;
border: 1px solid #fbbf24;
}
.btn-edit {
background: #3b82f6;
color: white;
}
.btn-edit:hover {
background: #2563eb;
}
.btn-delete {
background: #ef4444;
color: white;
}
.btn-delete:hover {
background: #dc2626;
}
.series {
color: #8b6f47;
font-size: 1rem;
margin: 0.5rem 0 1.5rem 0;
font-weight: 500;
font-style: italic;
}
.info-row {
display: flex;
align-items: center;
gap: 1rem;
margin-bottom: 0.75rem;
padding: 0.5rem 0;
border-bottom: 1px solid #f3f4f6;
}
.info-label {
font-weight: 600;
color: #4b5563;
min-width: 120px;
}
.info-value {
color: #1f2937;
}
.time-spent {
color: #10b981;
font-weight: 600;
}
.rating {
display: flex;
align-items: center;
gap: 0.25rem;
}
.rating span {
color: #e5e7eb;
font-size: 1.2rem;
}
.rating span.filled {
color: #f59e0b;
}
.rating-text {
margin-left: 0.5rem;
font-size: 1rem;
color: #6b7280;
font-weight: 500;
}
.like-section {
margin: 1.5rem 0;
}
.notes-section,
.tags-section,
.links-section {
margin-top: 2rem;
}
.notes-section h3,
.tags-section h3,
.links-section h3,
.comments-section h3 {
font-size: 1.25rem;
color: #1f2937;
margin-bottom: 1rem;
}
.notes {
font-size: 1rem;
color: #4b5563;
line-height: 1.6;
white-space: pre-wrap;
}
.tags-display {
display: flex;
flex-wrap: wrap;
gap: 0.5rem;
}
.tag-chip {
background: #8b6f47;
color: white;
padding: 0.375rem 0.875rem;
border-radius: 12px;
font-size: 0.875rem;
font-weight: 500;
}
.links-display {
display: flex;
flex-wrap: wrap;
gap: 0.75rem;
}
.external-link {
color: #8b6f47;
text-decoration: none;
font-size: 0.95rem;
padding: 0.5rem 1rem;
border: 2px solid #8b6f47;
border-radius: 4px;
transition: all 0.2s;
font-weight: 500;
}
.external-link:hover {
background: #8b6f47;
color: white;
}
.comments-section {
margin-top: 2rem;
padding-top: 2rem;
border-top: 2px solid #e5e7eb;
}
.comment-form {
margin-bottom: 1.5rem;
}
.comment-form textarea {
width: 100%;
padding: 0.75rem;
border: 1px solid #d1d5db;
border-radius: 4px;
font-size: 1rem;
resize: vertical;
margin-bottom: 0.75rem;
font-family: inherit;
}
.comment-form textarea:focus {
outline: none;
border-color: #8b6f47;
}
.banned-notice {
background: #fef2f2;
border: 1px solid #fecaca;
color: #991b1b;
padding: 1rem;
border-radius: 4px;
text-align: center;
margin-bottom: 1.5rem;
font-size: 0.95rem;
}
.auth-prompt {
background: #f3f4f6;
padding: 1rem;
border-radius: 4px;
text-align: center;
margin-bottom: 1.5rem;
}
.auth-prompt p {
margin: 0;
color: #4b5563;
}
.comments-loading {
text-align: center;
padding: 1.5rem;
color: #6b7280;
font-size: 0.95rem;
}
.btn {
padding: 0.625rem 1.25rem;
border: none;
border-radius: 4px;
cursor: pointer;
font-size: 1rem;
transition: all 0.3s;
font-weight: 500;
}
.btn:hover {
opacity: 0.9;
}
.btn-primary {
background: #8b6f47;
color: white;
}
@media (max-width: 640px) {
.container {
padding: 1rem;
}
.book-header {
flex-direction: column;
align-items: flex-start;
}
.book-header h1 {
font-size: 1.5rem;
}
.info-row {
flex-direction: column;
align-items: flex-start;
gap: 0.25rem;
}
.info-label {
min-width: auto;
}
}
`]
})
export class BookDetailComponent implements OnInit {
private readonly booksService = inject(BooksService);
private readonly commentsService = inject(CommentsService);
readonly authService = inject(AuthService);
private readonly sanitizeService = inject(SanitizeService);
private readonly route = inject(ActivatedRoute);
private readonly router = inject(Router);
book = signal<Book | null>(null);
comments = signal<Comment[]>([]);
loading = signal(true);
commentsLoading = signal(false);
error = signal<string | null>(null);
newCommentContent = '';
showEditForm = signal(false);
ngOnInit() {
const bookId = this.route.snapshot.paramMap.get('id');
if (!bookId) {
this.error.set('No book ID provided');
this.loading.set(false);
return;
}
this.loadBook(bookId);
this.loadComments(bookId);
}
private loadBook(bookId: string) {
this.loading.set(true);
this.booksService.getBookById(bookId).subscribe({
next: (book) => {
if (!book) {
this.error.set('Book not found');
} else {
this.book.set(book);
}
this.loading.set(false);
},
error: () => {
this.error.set('Failed to load book. It may not exist or there was an error.');
this.loading.set(false);
}
});
}
private loadComments(bookId: string) {
this.commentsLoading.set(true);
this.commentsService.getCommentsForBook(bookId).subscribe({
next: (comments) => {
this.comments.set(comments);
this.commentsLoading.set(false);
},
error: () => {
this.commentsLoading.set(false);
}
});
}
addComment() {
const book = this.book();
if (!book || !this.newCommentContent.trim()) return;
this.commentsService.addCommentToBook(book.id, { content: this.newCommentContent }).subscribe({
next: (comment) => {
this.comments.set([comment, ...this.comments()]);
this.newCommentContent = '';
}
});
}
handleCommentEdit(event: { commentId: string; content: string }) {
const book = this.book();
if (!book) return;
this.commentsService.updateCommentOnBook(book.id, event.commentId, event.content).subscribe({
next: (updatedComment) => {
this.comments.set(
this.comments().map(c => c.id === event.commentId ? updatedComment : c)
);
}
});
}
deleteComment(commentId: string) {
const book = this.book();
if (!book || !confirm('Are you sure you want to delete this comment?')) return;
this.commentsService.deleteCommentFromBook(book.id, commentId).subscribe({
next: () => {
this.comments.set(this.comments().filter(c => c.id !== commentId));
}
});
}
getStatusLabel(status: BookStatus): string {
switch (status) {
case BookStatus.reading: return 'Currently Reading';
case BookStatus.finished: return 'Finished';
case BookStatus.toRead: return 'To Read';
case BookStatus.retired: return 'Retired';
}
}
formatDate(date: Date | string): string {
return new Date(date).toLocaleDateString('en-GB', {
year: 'numeric',
month: 'long',
day: 'numeric'
});
}
formatTimeSpent(minutes: number): string {
const hours = Math.floor(minutes / 60);
const mins = minutes % 60;
if (hours === 0) {
return `${mins} minutes`;
} else if (mins === 0) {
return `${hours} hour${hours === 1 ? '' : 's'}`;
} else {
return `${hours} hour${hours === 1 ? '' : 's'} ${mins} minute${mins === 1 ? '' : 's'}`;
}
}
editBook() {
this.showEditForm.set(true);
}
cancelEdit() {
this.showEditForm.set(false);
}
saveEdit(data: UpdateBookDto) {
const book = this.book();
if (!book) return;
this.booksService.updateBook(book.id, data).subscribe({
next: (updatedBook) => {
this.book.set(updatedBook);
this.showEditForm.set(false);
},
error: (err) => {
alert('Failed to update book: ' + (err.error?.message || 'Unknown error'));
}
});
}
deleteBook() {
const book = this.book();
if (!book) return;
if (!confirm(`Are you sure you want to delete "${book.title}"? This action cannot be undone.`)) {
return;
}
this.booksService.deleteBook(book.id).subscribe({
next: () => {
this.router.navigate(['/books']);
},
error: (err) => {
alert('Failed to delete book: ' + (err.error?.message || 'Unknown error'));
}
});
}
}
@@ -7,7 +7,6 @@
import { Component, OnInit, inject, signal, computed } from '@angular/core';
import { CommonModule } from '@angular/common';
import { FormsModule } from '@angular/forms';
import { RouterLink } from '@angular/router';
import { BooksService } from '../../services/books.service';
import { AuthService } from '../../services/auth.service';
import { CommentsService } from '../../services/comments.service';
@@ -20,13 +19,9 @@ import { Book, BookStatus, CreateBookDto, UpdateBookDto, Comment, SuggestionEnti
@Component({
selector: 'app-books-list',
standalone: true,
imports: [CommonModule, FormsModule, RouterLink, PaginationComponent, LikeButtonComponent],
imports: [CommonModule, FormsModule, PaginationComponent, LikeButtonComponent],
template: `
<div class="container">
<div class="page-hero">
<img src="/assets/avatars/books-avatar.jpg" alt="Books avatar" class="page-avatar" />
</div>
<div class="header-section">
<h2>My Book Collection</h2>
@if (authService.isAdmin()) {
@@ -84,30 +79,9 @@ import { Book, BookStatus, CreateBookDto, UpdateBookDto, Comment, SuggestionEnti
<option [value]="BookStatus.reading">Currently Reading</option>
<option [value]="BookStatus.finished">Finished</option>
<option [value]="BookStatus.toRead">To Read</option>
<option [value]="BookStatus.retired">Retired</option>
</select>
</div>
<div class="form-group">
<label for="dateStarted">Date Started</label>
<input
type="date"
id="dateStarted"
[(ngModel)]="newBook.dateStarted"
name="dateStarted"
>
</div>
<div class="form-group">
<label for="dateFinished">Date Finished</label>
<input
type="date"
id="dateFinished"
[(ngModel)]="newBook.dateFinished"
name="dateFinished"
>
</div>
<div class="form-group">
<label for="rating">Rating (1-10)</label>
<input
@@ -120,34 +94,6 @@ import { Book, BookStatus, CreateBookDto, UpdateBookDto, Comment, SuggestionEnti
>
</div>
<div class="form-row">
<div class="form-group">
<label for="timeHours">Time Spent (Hours)</label>
<input
type="number"
id="timeHours"
[(ngModel)]="newBookTimeHours"
name="timeHours"
min="0"
placeholder="0"
(ngModelChange)="updateNewBookTimeSpent()"
>
</div>
<div class="form-group">
<label for="timeMinutes">Time Spent (Minutes)</label>
<input
type="number"
id="timeMinutes"
[(ngModel)]="newBookTimeMinutes"
name="timeMinutes"
min="0"
max="59"
placeholder="0"
(ngModelChange)="updateNewBookTimeSpent()"
>
</div>
</div>
<div class="form-group">
<label for="notes">Notes</label>
<textarea
@@ -159,29 +105,6 @@ import { Book, BookStatus, CreateBookDto, UpdateBookDto, Comment, SuggestionEnti
></textarea>
</div>
<div class="form-group">
<label for="series">Series (optional)</label>
<input
type="text"
id="series"
[(ngModel)]="newBook.series"
name="series"
placeholder="e.g., Harry Potter"
>
</div>
<div class="form-group">
<label for="seriesOrder">Series Order (optional)</label>
<input
type="number"
id="seriesOrder"
[(ngModel)]="newBook.seriesOrder"
name="seriesOrder"
min="1"
placeholder="Order in series"
>
</div>
<div class="form-group">
<label for="coverImage">Cover Image (max 500KB)</label>
<input
@@ -203,7 +126,8 @@ import { Book, BookStatus, CreateBookDto, UpdateBookDto, Comment, SuggestionEnti
</div>
<div class="form-group">
<div class="tags-input-container" aria-label="Tags">
<label>Tags</label>
<div class="tags-input-container">
@for (tag of newBook.tags; track tag; let i = $index) {
<span class="tag">
{{ tag }}
@@ -220,7 +144,8 @@ import { Book, BookStatus, CreateBookDto, UpdateBookDto, Comment, SuggestionEnti
</div>
</div>
<div class="form-group" aria-label="External Links">
<div class="form-group">
<label>External Links</label>
<div class="links-list">
@for (link of newBook.links; track link.url; let i = $index) {
<div class="link-item">
@@ -297,30 +222,9 @@ import { Book, BookStatus, CreateBookDto, UpdateBookDto, Comment, SuggestionEnti
<option [value]="BookStatus.reading">Currently Reading</option>
<option [value]="BookStatus.finished">Finished</option>
<option [value]="BookStatus.toRead">To Read</option>
<option [value]="BookStatus.retired">Retired</option>
</select>
</div>
<div class="form-group">
<label for="edit-dateStarted">Date Started</label>
<input
type="date"
id="edit-dateStarted"
[(ngModel)]="editBook.dateStarted"
name="dateStarted"
>
</div>
<div class="form-group">
<label for="edit-dateFinished">Date Finished</label>
<input
type="date"
id="edit-dateFinished"
[(ngModel)]="editBook.dateFinished"
name="dateFinished"
>
</div>
<div class="form-group">
<label for="edit-rating">Rating (1-10)</label>
<input
@@ -333,34 +237,6 @@ import { Book, BookStatus, CreateBookDto, UpdateBookDto, Comment, SuggestionEnti
>
</div>
<div class="form-row">
<div class="form-group">
<label for="edit-timeHours">Time Spent (Hours)</label>
<input
type="number"
id="edit-timeHours"
[(ngModel)]="editBookTimeHours"
name="timeHours"
min="0"
placeholder="0"
(ngModelChange)="updateEditBookTimeSpent()"
>
</div>
<div class="form-group">
<label for="edit-timeMinutes">Time Spent (Minutes)</label>
<input
type="number"
id="edit-timeMinutes"
[(ngModel)]="editBookTimeMinutes"
name="timeMinutes"
min="0"
max="59"
placeholder="0"
(ngModelChange)="updateEditBookTimeSpent()"
>
</div>
</div>
<div class="form-group">
<label for="edit-notes">Notes</label>
<textarea
@@ -372,29 +248,6 @@ import { Book, BookStatus, CreateBookDto, UpdateBookDto, Comment, SuggestionEnti
></textarea>
</div>
<div class="form-group">
<label for="edit-series">Series (optional)</label>
<input
type="text"
id="edit-series"
[(ngModel)]="editBook.series"
name="series"
placeholder="e.g., Harry Potter"
>
</div>
<div class="form-group">
<label for="edit-seriesOrder">Series Order (optional)</label>
<input
type="number"
id="edit-seriesOrder"
[(ngModel)]="editBook.seriesOrder"
name="seriesOrder"
min="1"
placeholder="Order in series"
>
</div>
<div class="form-group">
<label for="edit-coverImage">Cover Image (max 500KB)</label>
<input
@@ -416,7 +269,8 @@ import { Book, BookStatus, CreateBookDto, UpdateBookDto, Comment, SuggestionEnti
</div>
<div class="form-group">
<div class="tags-input-container" aria-label="Tags">
<label>Tags</label>
<div class="tags-input-container">
@for (tag of editBook.tags; track tag; let i = $index) {
<span class="tag">
{{ tag }}
@@ -433,7 +287,8 @@ import { Book, BookStatus, CreateBookDto, UpdateBookDto, Comment, SuggestionEnti
</div>
</div>
<div class="form-group" aria-label="External Links">
<div class="form-group">
<label>External Links</label>
<div class="links-list">
@for (link of editBook.links; track link.url; let i = $index) {
<div class="link-item">
@@ -566,7 +421,8 @@ import { Book, BookStatus, CreateBookDto, UpdateBookDto, Comment, SuggestionEnti
@if (showFilters()) {
<div class="advanced-filters">
<div class="filter-group">
<div class="tags-filter" aria-label="Filter by Tags">
<label>Filter by Tags:</label>
<div class="tags-filter">
@for (tag of allTags(); track tag) {
<label class="tag-checkbox">
<input
@@ -619,13 +475,6 @@ import { Book, BookStatus, CreateBookDto, UpdateBookDto, Comment, SuggestionEnti
>
To Read ({{ toReadCount() }})
</button>
<button
(click)="setFilter(BookStatus.retired)"
[class.active]="statusFilter() === BookStatus.retired"
class="filter-btn"
>
Retired ({{ retiredCount() }})
</button>
</div>
@if (loading()) {
@@ -646,35 +495,67 @@ import { Book, BookStatus, CreateBookDto, UpdateBookDto, Comment, SuggestionEnti
<div class="books-grid">
@for (book of paginatedBooks(); track book.id) {
<div class="book-card" [class.finished]="book.status === BookStatus.finished">
<a [routerLink]="['/books', book.id]" class="card-link">
<img [src]="book.coverImage || '/assets/default-cover.jpg'" [alt]="book.title" class="book-cover">
@if (book.coverImage) {
<img [src]="book.coverImage" [alt]="book.title" class="book-cover">
} @else {
<div class="book-cover placeholder">📚</div>
}
<div class="book-info">
<h3>{{ book.title }}</h3>
<p class="author">by {{ book.author }}</p>
<div class="book-info">
<h3>{{ book.title }}</h3>
<p class="author">by {{ book.author }}</p>
<span class="status status-{{ book.status }}">
{{ getStatusLabel(book.status) }}
</span>
<span class="status status-{{ book.status }}">
{{ getStatusLabel(book.status) }}
</span>
@if (book.rating) {
<div class="rating">
@for (star of [1,2,3,4,5,6,7,8,9,10]; track star) {
<span [class.filled]="star <= book.rating!">★</span>
}
</div>
}
</div>
</a>
@if (book.rating) {
<div class="rating">
@for (star of [1,2,3,4,5,6,7,8,9,10]; track star) {
<span [class.filled]="star <= book.rating!">★</span>
}
</div>
}
<div class="card-actions">
<app-like-button
entityType="book"
[entityId]="book.id"
></app-like-button>
@if (book.isbn) {
<p class="isbn">ISBN: {{ book.isbn }}</p>
}
@if (book.notes) {
<p class="notes">{{ book.notes }}</p>
}
@if (book.tags && book.tags.length > 0) {
<div class="tags-display">
@for (tag of book.tags; track tag) {
<span class="tag-chip">{{ tag }}</span>
}
</div>
}
@if (book.links && book.links.length > 0) {
<div class="links-display">
@for (link of book.links; track link.url) {
<a [href]="link.url" target="_blank" rel="noopener noreferrer" class="external-link">
{{ link.title }} ↗
</a>
}
</div>
}
@if (book.dateFinished) {
<p class="date-finished">
Finished: {{ formatDate(book.dateFinished) }}
</p>
}
@if (authService.isAdmin()) {
<div class="admin-actions">
<div class="actions">
<button (click)="startEdit(book)" class="btn btn-secondary btn-sm">
Edit
</button>
@@ -683,6 +564,85 @@ import { Book, BookStatus, CreateBookDto, UpdateBookDto, Comment, SuggestionEnti
</button>
</div>
}
<div class="comments-section">
<button (click)="toggleComments(book.id)" class="btn btn-secondary btn-sm comments-toggle">
{{ expandedComments()[book.id] ? 'Hide' : 'Show' }} Comments{{ comments()[book.id] ? ' (' + getCommentCount(book.id) + ')' : '' }}
</button>
@if (expandedComments()[book.id]) {
<div class="comments-container">
@if (authService.isAuthenticated()) {
@if (authService.user()?.isBanned) {
<div class="banned-notice">
You have been banned from commenting.
</div>
} @else {
<form (ngSubmit)="addComment(book.id)" class="comment-form">
<textarea
[(ngModel)]="newCommentContent[book.id]"
name="comment"
placeholder="Add a comment (Markdown supported)..."
rows="2"
></textarea>
<button type="submit" class="btn btn-primary btn-sm">Post Comment</button>
</form>
}
}
@if (commentsLoading()[book.id]) {
<div class="comments-loading">Loading comments...</div>
} @else {
@for (comment of comments()[book.id] || []; track comment.id) {
<div class="comment">
<div class="comment-header">
@if (comment.user.avatar) {
<img [src]="comment.user.avatar" [alt]="comment.user.username" class="comment-avatar">
}
<span class="comment-author">{{ comment.user.username }}</span>
@if (comment.user.inDiscord) {
<span class="discord-badge">Discord</span>
}
@if (comment.user.isVip) {
<span class="vip-badge">VIP</span>
}
@if (comment.user.isMod) {
<span class="mod-badge">Mod</span>
}
@if (comment.user.isStaff) {
<span class="staff-badge">Staff</span>
}
<span class="comment-date">{{ formatDate(comment.createdAt) }}</span>
@if (canEditComment(comment)) {
<button (click)="startEditComment(book.id, comment)" class="btn btn-secondary btn-xs">Edit</button>
}
@if (canDeleteComment(comment)) {
<button (click)="deleteComment(book.id, comment.id)" class="btn btn-danger btn-xs">Delete</button>
}
</div>
@if (editingCommentId() === comment.id) {
<div class="comment-edit-form">
<textarea
[(ngModel)]="editCommentContent"
name="editComment"
rows="3"
></textarea>
<div class="comment-edit-actions">
<button (click)="saveCommentEdit(book.id, comment.id)" class="btn btn-primary btn-xs">Save</button>
<button (click)="cancelCommentEdit()" class="btn btn-secondary btn-xs">Cancel</button>
</div>
</div>
} @else {
<div class="comment-content" [innerHTML]="sanitizeService.sanitizeHtml(comment.content)"></div>
}
</div>
} @empty {
<div class="no-comments">No comments yet. Be the first to comment!</div>
}
}
</div>
}
</div>
</div>
</div>
}
@@ -705,26 +665,6 @@ import { Book, BookStatus, CreateBookDto, UpdateBookDto, Comment, SuggestionEnti
padding: 2rem;
}
.page-hero {
text-align: center;
margin-bottom: 2rem;
}
.page-avatar {
width: 120px;
height: 120px;
border-radius: 50%;
object-fit: cover;
border: 3px solid #8b6f47;
box-shadow: 0 4px 12px rgba(139, 111, 71, 0.3);
transition: all 0.3s;
}
.page-avatar:hover {
transform: scale(1.05);
box-shadow: 0 8px 16px rgba(139, 111, 71, 0.5);
}
.header-section {
display: flex;
justify-content: space-between;
@@ -757,13 +697,6 @@ import { Book, BookStatus, CreateBookDto, UpdateBookDto, Comment, SuggestionEnti
font-style: italic;
}
.form-row {
display: grid;
grid-template-columns: 1fr 1fr;
gap: 1rem;
margin-bottom: 1rem;
}
.form-group {
margin-bottom: 1rem;
}
@@ -968,8 +901,6 @@ import { Book, BookStatus, CreateBookDto, UpdateBookDto, Comment, SuggestionEnti
overflow: hidden;
transition: all 0.3s;
backdrop-filter: blur(10px);
display: flex;
flex-direction: column;
}
.book-card:hover {
@@ -983,33 +914,6 @@ import { Book, BookStatus, CreateBookDto, UpdateBookDto, Comment, SuggestionEnti
border-color: var(--witch-mauve);
}
.card-link {
text-decoration: none;
color: inherit;
flex: 1;
display: flex;
flex-direction: column;
}
.card-link:hover h3 {
color: var(--witch-rose);
}
.card-actions {
padding: 1rem;
border-top: 1px solid var(--witch-lavender);
display: flex;
gap: 0.5rem;
align-items: center;
flex-wrap: wrap;
}
.admin-actions {
display: flex;
gap: 0.5rem;
margin-left: auto;
}
.book-cover {
width: 100%;
height: 250px;
@@ -1031,7 +935,6 @@ import { Book, BookStatus, CreateBookDto, UpdateBookDto, Comment, SuggestionEnti
.book-info h3 {
margin: 0 0 0.5rem 0;
font-size: 1.1rem;
transition: color 0.2s;
}
.author {
@@ -1074,6 +977,28 @@ import { Book, BookStatus, CreateBookDto, UpdateBookDto, Comment, SuggestionEnti
color: var(--witch-rose);
}
.isbn {
font-size: 0.8rem;
color: var(--witch-mauve);
margin: 0.5rem 0;
}
.notes {
font-size: 0.9rem;
color: var(--witch-plum);
margin: 0.5rem 0;
}
.date-finished {
font-size: 0.85rem;
color: var(--witch-plum);
margin-top: 0.5rem;
}
.actions {
margin-top: 1rem;
}
.btn {
padding: 0.5rem 1rem;
border: none;
@@ -1120,6 +1045,163 @@ import { Book, BookStatus, CreateBookDto, UpdateBookDto, Comment, SuggestionEnti
.btn-sm { padding: 0.25rem 0.75rem; font-size: 0.85rem; }
.btn-xs { padding: 0.15rem 0.5rem; font-size: 0.75rem; }
.comments-section {
margin-top: 1rem;
border-top: 1px solid var(--witch-lavender);
padding-top: 1rem;
}
.comments-toggle {
width: 100%;
}
.comments-container {
margin-top: 1rem;
}
.comment-form {
margin-bottom: 1rem;
}
.comment-form textarea {
width: 100%;
padding: 0.5rem;
border: 2px solid var(--witch-lavender);
border-radius: 4px;
font-size: 0.9rem;
background-color: var(--witch-moon);
color: var(--witch-purple);
resize: vertical;
margin-bottom: 0.5rem;
}
.comment-form textarea:focus {
outline: none;
border-color: var(--witch-rose);
}
.comment {
background: var(--witch-moon);
border: 1px solid var(--witch-lavender);
border-radius: 4px;
padding: 0.75rem;
margin-bottom: 0.5rem;
}
.comment-header {
display: flex;
align-items: center;
gap: 0.5rem;
margin-bottom: 0.5rem;
flex-wrap: wrap;
}
.comment-avatar {
width: 24px;
height: 24px;
border-radius: 50%;
}
.comment-author {
font-weight: 500;
color: var(--witch-plum);
}
.discord-badge {
background: #5865f2;
color: white;
padding: 0.1rem 0.4rem;
border-radius: 3px;
font-size: 0.7rem;
font-weight: 500;
}
.vip-badge {
background: linear-gradient(135deg, #ffd700, #ffaa00);
color: #1a1a1a;
padding: 0.1rem 0.4rem;
border-radius: 3px;
font-size: 0.7rem;
font-weight: 600;
}
.mod-badge {
background: linear-gradient(135deg, #00b894, #00cec9);
color: white;
padding: 0.1rem 0.4rem;
border-radius: 3px;
font-size: 0.7rem;
font-weight: 600;
}
.staff-badge {
background: linear-gradient(135deg, #e84393, #fd79a8);
color: white;
padding: 0.1rem 0.4rem;
border-radius: 3px;
font-size: 0.7rem;
font-weight: 600;
}
.comment-date {
font-size: 0.75rem;
color: var(--witch-mauve);
}
.comment-content {
font-size: 0.9rem;
color: var(--witch-purple);
}
.comment-content :deep(p) {
margin: 0.25rem 0;
}
.comments-loading,
.no-comments {
text-align: center;
padding: 1rem;
color: var(--witch-mauve);
font-size: 0.9rem;
}
.banned-notice {
background: #fef2f2;
border: 1px solid #fecaca;
color: #991b1b;
padding: 0.75rem 1rem;
border-radius: 4px;
text-align: center;
margin-bottom: 1rem;
font-size: 0.9rem;
}
.comment-edit-form {
margin-top: 0.5rem;
}
.comment-edit-form textarea {
width: 100%;
padding: 0.5rem;
border: 2px solid var(--witch-lavender);
border-radius: 4px;
font-size: 0.9rem;
background-color: var(--witch-moon);
color: var(--witch-purple);
resize: vertical;
}
.comment-edit-form textarea:focus {
outline: none;
border-color: var(--witch-rose);
}
.comment-edit-actions {
display: flex;
gap: 0.5rem;
margin-top: 0.5rem;
}
.image-preview {
margin-top: 0.5rem;
display: flex;
@@ -1201,6 +1283,21 @@ import { Book, BookStatus, CreateBookDto, UpdateBookDto, Comment, SuggestionEnti
opacity: 0.8;
}
.tags-display {
display: flex;
flex-wrap: wrap;
gap: 0.5rem;
margin: 0.5rem 0;
}
.tag-chip {
background: rgba(139, 111, 71, 0.2);
color: #8b6f47;
padding: 0.2rem 0.5rem;
border-radius: 4px;
font-size: 0.8rem;
}
.links-list {
margin-bottom: 0.5rem;
}
@@ -1226,6 +1323,28 @@ import { Book, BookStatus, CreateBookDto, UpdateBookDto, Comment, SuggestionEnti
min-width: 120px;
}
.links-display {
display: flex;
flex-wrap: wrap;
gap: 0.5rem;
margin: 0.5rem 0;
}
.external-link {
color: #8b6f47;
text-decoration: none;
font-size: 0.85rem;
padding: 0.2rem 0.5rem;
background: rgba(139, 111, 71, 0.1);
border-radius: 4px;
transition: background 0.2s;
}
.external-link:hover {
background: rgba(139, 111, 71, 0.2);
text-decoration: underline;
}
@media (max-width: 768px) {
.search-section {
flex-direction: column;
@@ -1298,7 +1417,6 @@ export class BooksListComponent implements OnInit {
readingCount = computed(() => this.books().filter(book => book.status === BookStatus.reading).length);
finishedCount = computed(() => this.books().filter(book => book.status === BookStatus.finished).length);
toReadCount = computed(() => this.books().filter(book => book.status === BookStatus.toRead).length);
retiredCount = computed(() => this.books().filter(book => book.status === BookStatus.retired).length);
// Get all unique tags from all books
allTags = computed(() => {
@@ -1349,13 +1467,11 @@ export class BooksListComponent implements OnInit {
totalFilteredBooks = computed(() => this.filteredBooks().length);
newBook: Partial<CreateBookDto> & { dateStarted?: Date; dateFinished?: Date } = {
newBook: Partial<CreateBookDto> = {
title: '',
author: '',
isbn: '',
status: BookStatus.toRead,
dateStarted: undefined,
dateFinished: undefined,
rating: undefined,
notes: '',
tags: [],
@@ -1364,11 +1480,6 @@ export class BooksListComponent implements OnInit {
editBook: Partial<UpdateBookDto> = {};
newBookTimeHours = 0;
newBookTimeMinutes = 0;
editBookTimeHours = 0;
editBookTimeMinutes = 0;
// Tags and links input state
newTagInput = '';
editTagInput = '';
@@ -1441,7 +1552,6 @@ export class BooksListComponent implements OnInit {
case BookStatus.reading: return 'Currently Reading';
case BookStatus.finished: return 'Finished';
case BookStatus.toRead: return 'To Read';
case BookStatus.retired: return 'Retired';
}
}
@@ -1458,8 +1568,6 @@ export class BooksListComponent implements OnInit {
author: '',
isbn: '',
status: BookStatus.toRead,
dateStarted: undefined,
dateFinished: undefined,
rating: undefined,
notes: '',
coverImage: undefined,
@@ -1471,8 +1579,6 @@ export class BooksListComponent implements OnInit {
this.newTagInput = '';
this.newLinkTitle = '';
this.newLinkUrl = '';
this.newBookTimeHours = 0;
this.newBookTimeMinutes = 0;
}
addTag(target: 'new' | 'edit') {
@@ -1528,8 +1634,6 @@ export class BooksListComponent implements OnInit {
author: this.newBook.author,
isbn: this.newBook.isbn,
status: this.newBook.status,
dateStarted: this.newBook.dateStarted ? new Date(this.newBook.dateStarted) : undefined,
dateFinished: this.newBook.dateFinished ? new Date(this.newBook.dateFinished) : undefined,
rating: this.newBook.rating,
notes: this.newBook.notes,
coverImage: this.newBook.coverImage,
@@ -1558,15 +1662,11 @@ export class BooksListComponent implements OnInit {
author: book.author,
isbn: book.isbn,
status: book.status,
dateStarted: book.dateStarted,
dateFinished: book.dateFinished,
rating: book.rating,
notes: book.notes,
coverImage: book.coverImage,
tags: [...(book.tags || [])],
links: [...(book.links || [])],
series: book.series,
seriesOrder: book.seriesOrder
links: [...(book.links || [])]
};
this.editBookImagePreview.set(book.coverImage || null);
this.showAddForm.set(false);
@@ -1574,16 +1674,6 @@ export class BooksListComponent implements OnInit {
this.editTagInput = '';
this.editLinkTitle = '';
this.editLinkUrl = '';
if (book.timeSpent) {
this.editBookTimeHours = Math.floor(book.timeSpent / 60);
this.editBookTimeMinutes = book.timeSpent % 60;
} else {
this.editBookTimeHours = 0;
this.editBookTimeMinutes = 0;
}
// Scroll to top so the form is visible
window.scrollTo({ top: 0, behavior: 'smooth' });
}
cancelEdit() {
@@ -1600,13 +1690,7 @@ export class BooksListComponent implements OnInit {
const book = this.editingBook();
if (!book || !this.editBook.title || !this.editBook.author || !this.editBook.status) return;
const updateData = {
...this.editBook,
dateStarted: this.editBook.dateStarted ? new Date(this.editBook.dateStarted) : undefined,
dateFinished: this.editBook.dateFinished ? new Date(this.editBook.dateFinished) : undefined,
};
this.booksService.updateBook(book.id, updateData).subscribe(() => {
this.booksService.updateBook(book.id, this.editBook).subscribe(() => {
this.loadBooks();
this.cancelEdit();
});
@@ -1616,29 +1700,6 @@ export class BooksListComponent implements OnInit {
return new Date(date).toLocaleDateString();
}
updateNewBookTimeSpent() {
const totalMinutes = (this.newBookTimeHours * 60) + this.newBookTimeMinutes;
this.newBook.timeSpent = totalMinutes > 0 ? totalMinutes : undefined;
}
updateEditBookTimeSpent() {
const totalMinutes = (this.editBookTimeHours * 60) + this.editBookTimeMinutes;
this.editBook.timeSpent = totalMinutes > 0 ? totalMinutes : undefined;
}
formatTimeSpent(minutes: number): string {
const hours = Math.floor(minutes / 60);
const mins = minutes % 60;
if (hours === 0) {
return `${mins}m`;
} else if (mins === 0) {
return `${hours}h`;
} else {
return `${hours}h ${mins}m`;
}
}
// Image handling methods
onImageSelected(event: Event, target: 'new' | 'edit' | 'suggest') {
const input = event.target as HTMLInputElement;
@@ -1827,7 +1888,7 @@ export class BooksListComponent implements OnInit {
try {
await this.suggestionService.createSuggestion({
entityType: SuggestionEntity.book,
entityType: SuggestionEntity.BOOK,
title: this.suggestedBook.title,
author: this.suggestedBook.author,
isbn: this.suggestedBook.isbn,
@@ -1840,21 +1901,4 @@ export class BooksListComponent implements OnInit {
alert('Failed to submit suggestion. Please try again.');
}
}
handleCommentEdit(bookId: string, event: { commentId: string; content: string }) {
this.commentsService.updateCommentOnBook(bookId, event.commentId, event.content).subscribe({
next: (updatedComment) => {
this.comments.set({
...this.comments(),
[bookId]: (this.comments()[bookId] || []).map(c =>
c.id === event.commentId ? updatedComment : c
)
});
}
});
}
getCommentsSignal(bookId: string) {
return signal(this.comments()[bookId] || []);
}
}
@@ -1,317 +0,0 @@
/**
* @copyright NHCarrigan
* @license Naomi's Public License
* @author Naomi Carrigan
*/
import { Component, Input, Output, EventEmitter, signal, inject } from '@angular/core';
import { CommonModule } from '@angular/common';
import { FormsModule } from '@angular/forms';
import type { Comment } from '@library/shared-types';
import { PrimaryBadge } from '@library/shared-types';
import { AuthService } from '../../services/auth.service';
import { SanitizeService } from '../../services/sanitize.service';
import { ReportModalComponent } from '../report-modal/report-modal.component';
@Component({
selector: 'app-comment-display',
standalone: true,
imports: [CommonModule, FormsModule, ReportModalComponent],
template: `
<div class="comments-section">
@if (comments().length > 0) {
@for (comment of comments(); track comment.id) {
<div class="comment">
<div class="comment-header">
@if (comment.user.avatar) {
<img [src]="comment.user.avatar" [alt]="comment.user.username" class="comment-avatar">
}
<span class="comment-author">{{ comment.user.username }}</span>
@if (comment.user.primaryBadge) {
<!-- Show only the selected primary badge -->
@if (comment.user.primaryBadge === PrimaryBadge.STAFF && comment.user.isStaff) {
<span class="staff-badge">Staff</span>
}
@if (comment.user.primaryBadge === PrimaryBadge.MOD && comment.user.isMod) {
<span class="mod-badge">Mod</span>
}
@if (comment.user.primaryBadge === PrimaryBadge.VIP && comment.user.isVip) {
<span class="vip-badge">VIP</span>
}
@if (comment.user.primaryBadge === PrimaryBadge.DISCORD && comment.user.inDiscord) {
<span class="discord-badge">Discord</span>
}
}
<span class="comment-date">{{ formatDate(comment.createdAt) }}</span>
@if (canEditComment(comment)) {
<button (click)="startEdit(comment)" class="btn btn-secondary btn-xs">Edit</button>
}
@if (canDeleteComment(comment)) {
<button (click)="delete.emit(comment.id)" class="btn btn-danger btn-xs">Delete</button>
}
@if (canReportComment(comment)) {
<button (click)="openReportModal(comment)" class="btn btn-warning btn-xs">Report</button>
}
</div>
@if (editingCommentId() === comment.id) {
<div class="comment-edit-form">
<textarea
[(ngModel)]="editCommentContent"
name="editComment"
rows="3"
></textarea>
<div class="comment-edit-actions">
<button (click)="saveEdit(comment.id)" class="btn btn-primary btn-xs">Save</button>
<button (click)="cancelEdit()" class="btn btn-secondary btn-xs">Cancel</button>
</div>
</div>
} @else {
@if (comment.hasPendingReports) {
<div class="comment-pending-review">[comment pending admin review]</div>
} @else {
<div class="comment-content" [innerHTML]="sanitizeService.sanitizeHtml(comment.content)"></div>
}
}
</div>
}
} @else {
<div class="no-comments">No comments yet. Be the first to comment!</div>
}
</div>
@if (showReportModal()) {
<app-report-modal
[reportType]="'comment'"
[targetId]="reportingCommentId()"
(closeModal)="closeReportModal()"
/>
}
`,
styles: [`
.comments-section {
margin-top: 1rem;
}
.comment {
background: #f9fafb;
padding: 0.75rem;
border-radius: 0.375rem;
margin-bottom: 0.75rem;
}
.comment-header {
display: flex;
align-items: center;
gap: 0.5rem;
margin-bottom: 0.5rem;
flex-wrap: wrap;
}
.comment-avatar {
width: 2rem;
height: 2rem;
border-radius: 50%;
object-fit: cover;
}
.comment-author {
font-weight: 600;
color: #1f2937;
}
.discord-badge,
.vip-badge,
.mod-badge,
.staff-badge {
padding: 0.125rem 0.5rem;
border-radius: 0.25rem;
font-size: 0.75rem;
font-weight: 600;
}
.discord-badge {
background: #5865f2;
color: white;
}
.vip-badge {
background: #fbbf24;
color: #78350f;
}
.mod-badge {
background: #10b981;
color: white;
}
.staff-badge {
background: #8b5cf6;
color: white;
}
.comment-date {
font-size: 0.75rem;
color: #6b7280;
}
.comment-content {
font-size: 0.9rem;
color: #4b5563;
}
.comment-pending-review {
font-size: 0.9rem;
color: #9b59b6;
font-style: italic;
padding: 0.5rem;
background: #f3e8ff;
border-radius: 0.25rem;
}
.comment-edit-form {
margin-top: 0.5rem;
}
.comment-edit-form textarea {
width: 100%;
padding: 0.5rem;
border: 1px solid #d1d5db;
border-radius: 0.375rem;
font-family: inherit;
resize: vertical;
}
.comment-edit-actions {
display: flex;
gap: 0.5rem;
margin-top: 0.5rem;
}
.no-comments {
text-align: center;
color: #6b7280;
padding: 2rem;
font-style: italic;
}
.btn {
padding: 0.25rem 0.75rem;
border: none;
border-radius: 0.25rem;
cursor: pointer;
font-size: 0.875rem;
transition: all 0.2s;
}
.btn-xs {
padding: 0.125rem 0.5rem;
font-size: 0.75rem;
}
.btn-primary {
background: #3b82f6;
color: white;
}
.btn-primary:hover {
background: #2563eb;
}
.btn-secondary {
background: #6b7280;
color: white;
}
.btn-secondary:hover {
background: #4b5563;
}
.btn-danger {
background: #ef4444;
color: white;
}
.btn-danger:hover {
background: #dc2626;
}
.btn-warning {
background: #f59e0b;
color: white;
}
.btn-warning:hover {
background: #d97706;
}
`]
})
export class CommentDisplayComponent {
private readonly authService = inject(AuthService);
readonly sanitizeService = inject(SanitizeService);
// Expose PrimaryBadge enum for template
readonly PrimaryBadge = PrimaryBadge;
@Input({ required: true }) comments = signal<Comment[]>([]);
@Output() edit = new EventEmitter<{ commentId: string; content: string }>();
@Output() delete = new EventEmitter<string>();
editingCommentId = signal<string | null>(null);
editCommentContent = '';
showReportModal = signal(false);
reportingCommentId = signal<string>('');
formatDate(date: Date | string): string {
const d = new Date(date);
return d.toLocaleDateString('en-GB', {
year: 'numeric',
month: 'short',
day: 'numeric',
hour: '2-digit',
minute: '2-digit',
});
}
canEditComment(comment: Comment): boolean {
const user = this.authService.user();
if (!user) return false;
return comment.userId === user.id || this.authService.isAdmin();
}
canDeleteComment(comment: Comment): boolean {
const user = this.authService.user();
if (!user) return false;
return comment.userId === user.id || this.authService.isAdmin();
}
canReportComment(comment: Comment): boolean {
const user = this.authService.user();
if (!user) return false;
// Users can report comments they didn't write (but not their own)
return comment.userId !== user.id;
}
startEdit(comment: Comment): void {
this.editingCommentId.set(comment.id);
this.editCommentContent = comment.rawContent ?? comment.content;
}
saveEdit(commentId: string): void {
this.edit.emit({ commentId, content: this.editCommentContent });
this.cancelEdit();
}
cancelEdit(): void {
this.editingCommentId.set(null);
this.editCommentContent = '';
}
openReportModal(comment: Comment): void {
this.reportingCommentId.set(comment.id);
this.showReportModal.set(true);
}
closeReportModal(): void {
this.showReportModal.set(false);
this.reportingCommentId.set('');
}
}
@@ -12,24 +12,24 @@ import { CommonModule } from '@angular/common';
standalone: true,
imports: [CommonModule],
template: `
<footer class="footer" role="contentinfo">
<footer class="footer">
<div class="footer-content">
<div class="cta-section">
<section class="cta-card discord" aria-labelledby="discord-heading">
<h2 id="discord-heading">Join the Community</h2>
<div class="cta-card discord">
<h3>Join the Community</h3>
<p>Want to chat about what we're reading, playing, or listening to? Got recommendations? Come hang out!</p>
<a href="https://chat.nhcarrigan.com" target="_blank" rel="noopener noreferrer" class="btn btn-discord">
Join our Discord<span class="sr-only"> (opens in new window)</span>
Join our Discord
</a>
</section>
</div>
<section class="cta-card donate" aria-labelledby="donate-heading">
<h2 id="donate-heading">Support Naomi</h2>
<div class="cta-card donate">
<h3>Support Naomi</h3>
<p>Enjoying the vibes? Your support helps keep the servers running and the coffee flowing!</p>
<a href="https://donate.nhcarrigan.com" target="_blank" rel="noopener noreferrer" class="btn btn-donate">
Buy us a coffee<span class="sr-only"> (opens in new window)</span>
Buy us a coffee
</a>
</section>
</div>
</div>
<div class="footer-bottom">
@@ -73,7 +73,7 @@ import { CommonModule } from '@angular/common';
box-shadow: 0 8px 24px rgba(0, 0, 0, 0.3);
}
.cta-card h2 {
.cta-card h3 {
margin: 0 0 0.75rem 0;
font-size: 1.25rem;
color: var(--witch-moon);
@@ -131,18 +131,6 @@ import { CommonModule } from '@angular/common';
font-size: 0.9rem;
color: var(--witch-lavender);
}
.sr-only {
position: absolute;
width: 1px;
height: 1px;
padding: 0;
margin: -1px;
overflow: hidden;
clip: rect(0, 0, 0, 0);
white-space: nowrap;
border-width: 0;
}
`]
})
export class FooterComponent {
@@ -1,730 +0,0 @@
/**
* @copyright 2026 NHCarrigan
* @license Naomi's Public License
* @author Naomi Carrigan
*/
import { Component, OnInit, inject, signal } from '@angular/core';
import { CommonModule } from '@angular/common';
import { RouterLink, ActivatedRoute, Router } from '@angular/router';
import { FormsModule } from '@angular/forms';
import { GamesService } from '../../services/games.service';
import { CommentsService } from '../../services/comments.service';
import { AuthService } from '../../services/auth.service';
import { SanitizeService } from '../../services/sanitize.service';
import { CommentDisplayComponent } from '../comment-display/comment-display.component';
import { LikeButtonComponent } from '../shared/like-button.component';
import { GameFormComponent } from '../shared/game-form.component';
import { Game, Comment, GameStatus, UpdateGameDto } from '@library/shared-types';
@Component({
selector: 'app-game-detail',
standalone: true,
imports: [CommonModule, RouterLink, FormsModule, CommentDisplayComponent, LikeButtonComponent, GameFormComponent],
template: `
<div class="container">
<div class="breadcrumb">
<a routerLink="/games" class="breadcrumb-link">← Back to Games</a>
</div>
@if (loading()) {
<div class="loading">Loading game details...</div>
} @else if (error()) {
<div class="error-state">
<h2>Game Not Found</h2>
<p>{{ error() }}</p>
<a routerLink="/games" class="btn btn-primary">Return to Games</a>
</div>
} @else if (game()) {
<div class="game-detail-card">
<div class="game-cover-section">
<img [src]="game()!.coverImage || '/assets/default-cover.jpg'" [alt]="game()!.title" class="game-cover-large">
</div>
<div class="game-content">
<div class="game-header">
<h1>{{ game()!.title }}</h1>
<span class="status status-{{ game()!.status }}">
{{ getStatusLabel(game()!.status) }}
</span>
</div>
@if (authService.user()?.isAdmin) {
<div class="admin-actions">
<button (click)="toggleEditForm()" class="btn btn-edit">✏️ {{ showEditForm() ? 'Cancel Edit' : 'Edit' }}</button>
<button (click)="deleteGame()" class="btn btn-delete">🗑️ Delete</button>
</div>
}
@if (showEditForm() && authService.user()?.isAdmin && game()) {
<app-game-form
mode="edit"
[game]="game()!"
(formSubmit)="saveEdit($event)"
(formCancel)="cancelEdit()"
></app-game-form>
}
@if (game()!.series) {
<p class="series">
📚 {{ game()!.series }}@if (game()!.seriesOrder) { #{{ game()!.seriesOrder }}}
</p>
}
@if (game()!.platform) {
<div class="info-row">
<span class="info-label">Platform:</span>
<span class="info-value">{{ game()!.platform }}</span>
</div>
}
@if (game()!.rating) {
<div class="info-row">
<span class="info-label">Rating:</span>
<div class="rating">
@for (star of [1,2,3,4,5,6,7,8,9,10]; track star) {
<span [class.filled]="star <= game()!.rating!">★</span>
}
<span class="rating-text">({{ game()!.rating }}/10)</span>
</div>
</div>
}
@if (game()!.timeSpent) {
<div class="info-row">
<span class="info-label">Time Played:</span>
<span class="info-value time-spent">{{ formatTimeSpent(game()!.timeSpent!) }}</span>
</div>
}
@if (game()!.dateStarted) {
<div class="info-row">
<span class="info-label">Started:</span>
<span class="info-value">{{ formatDate(game()!.dateStarted!) }}</span>
</div>
}
@if (game()!.dateFinished) {
<div class="info-row">
<span class="info-label">Finished:</span>
<span class="info-value">{{ formatDate(game()!.dateFinished!) }}</span>
</div>
}
<div class="info-row">
<span class="info-label">Added:</span>
<span class="info-value">{{ formatDate(game()!.createdAt) }}</span>
</div>
<div class="info-row">
<span class="info-label">Last Updated:</span>
<span class="info-value">{{ formatDate(game()!.updatedAt) }}</span>
</div>
<div class="like-section">
<app-like-button
entityType="game"
[entityId]="game()!.id"
></app-like-button>
</div>
@if (game()!.notes) {
<div class="notes-section">
<h3>Notes</h3>
<p class="notes">{{ game()!.notes }}</p>
</div>
}
@if (game()!.tags && game()!.tags.length > 0) {
<div class="tags-section">
<h3>Tags</h3>
<div class="tags-display">
@for (tag of game()!.tags; track tag) {
<span class="tag-chip">{{ tag }}</span>
}
</div>
</div>
}
@if (game()!.links && game()!.links.length > 0) {
<div class="links-section">
<h3>External Links</h3>
<div class="links-display">
@for (link of game()!.links; track link.url) {
<a [href]="link.url" target="_blank" rel="noopener noreferrer" class="external-link">
{{ link.title }} ↗
</a>
}
</div>
</div>
}
<div class="comments-section">
<h3>Comments</h3>
@if (authService.isAuthenticated()) {
@if (authService.user()?.isBanned) {
<div class="banned-notice">
You have been banned from commenting.
</div>
} @else {
<form (ngSubmit)="addComment()" class="comment-form">
<textarea
[(ngModel)]="newCommentContent"
name="comment"
placeholder="Add a comment (Markdown supported)..."
rows="3"
></textarea>
<button type="submit" class="btn btn-primary">Post Comment</button>
</form>
}
} @else {
<div class="auth-prompt">
<p>Please sign in to comment.</p>
</div>
}
@if (commentsLoading()) {
<div class="comments-loading">Loading comments...</div>
} @else {
<app-comment-display
[comments]="comments"
(edit)="handleCommentEdit($event)"
(delete)="deleteComment($event)"
/>
}
</div>
</div>
</div>
}
</div>
`,
styles: [`
.container {
max-width: 900px;
margin: 0 auto;
padding: 2rem;
}
.breadcrumb {
margin-bottom: 1.5rem;
}
.breadcrumb-link {
color: #ff6b6b;
text-decoration: none;
font-weight: 500;
transition: opacity 0.3s;
}
.breadcrumb-link:hover {
opacity: 0.8;
}
.loading {
text-align: center;
padding: 3rem;
color: #666;
font-size: 1.1rem;
}
.error-state {
text-align: center;
padding: 3rem;
background: #fef2f2;
border: 1px solid #fecaca;
border-radius: 8px;
}
.error-state h2 {
color: #991b1b;
margin-bottom: 1rem;
}
.error-state p {
color: #dc2626;
margin-bottom: 1.5rem;
}
.game-detail-card {
background: white;
border: 1px solid #e5e7eb;
border-radius: 8px;
overflow: hidden;
box-shadow: 0 1px 3px rgba(0,0,0,0.1);
}
.game-cover-section {
width: 100%;
background: #f3f4f6;
display: flex;
justify-content: center;
align-items: center;
padding: 2rem;
}
.game-cover-large {
max-width: 100%;
max-height: 400px;
object-fit: contain;
border-radius: 4px;
}
.game-content {
padding: 2rem;
}
.game-header {
display: flex;
align-items: center;
gap: 1rem;
margin-bottom: 1rem;
flex-wrap: wrap;
}
.game-header h1 {
margin: 0;
font-size: 2rem;
color: #1f2937;
}
.status {
display: inline-block;
padding: 0.5rem 1rem;
border-radius: 4px;
font-size: 0.9rem;
font-weight: 500;
}
.status-playing {
background: #fef3c7;
color: #92400e;
}
.status-completed {
background: #d1fae5;
color: #065f46;
}
.status-backlog {
background: #e0e7ff;
color: #3730a3;
}
.status-retired {
background: #f3f4f6;
color: #4b5563;
}
.admin-actions {
display: flex;
gap: 0.75rem;
margin-bottom: 1.5rem;
padding: 1rem;
background: #fef3c7;
border-radius: 4px;
border: 1px solid #fbbf24;
}
.btn-edit {
background: #3b82f6;
color: white;
}
.btn-edit:hover {
background: #2563eb;
}
.btn-delete {
background: #ef4444;
color: white;
}
.btn-delete:hover {
background: #dc2626;
}
.series {
color: #8b6f47;
font-size: 1rem;
margin: 0.5rem 0 1.5rem 0;
font-weight: 500;
font-style: italic;
}
.info-row {
display: flex;
align-items: center;
gap: 1rem;
margin-bottom: 0.75rem;
padding: 0.5rem 0;
border-bottom: 1px solid #f3f4f6;
}
.info-label {
font-weight: 600;
color: #4b5563;
min-width: 120px;
}
.info-value {
color: #1f2937;
}
.time-spent {
color: #10b981;
font-weight: 600;
}
.rating {
display: flex;
align-items: center;
gap: 0.25rem;
}
.rating span {
color: #e5e7eb;
font-size: 1.2rem;
}
.rating span.filled {
color: #f59e0b;
}
.rating-text {
margin-left: 0.5rem;
font-size: 1rem;
color: #6b7280;
font-weight: 500;
}
.like-section {
margin: 1.5rem 0;
}
.notes-section,
.tags-section,
.links-section {
margin-top: 2rem;
}
.notes-section h3,
.tags-section h3,
.links-section h3,
.comments-section h3 {
font-size: 1.25rem;
color: #1f2937;
margin-bottom: 1rem;
}
.notes {
font-size: 1rem;
color: #4b5563;
line-height: 1.6;
white-space: pre-wrap;
}
.tags-display {
display: flex;
flex-wrap: wrap;
gap: 0.5rem;
}
.tag-chip {
background: #ff6b6b;
color: white;
padding: 0.375rem 0.875rem;
border-radius: 12px;
font-size: 0.875rem;
font-weight: 500;
}
.links-display {
display: flex;
flex-wrap: wrap;
gap: 0.75rem;
}
.external-link {
color: #ff6b6b;
text-decoration: none;
font-size: 0.95rem;
padding: 0.5rem 1rem;
border: 2px solid #ff6b6b;
border-radius: 4px;
transition: all 0.2s;
font-weight: 500;
}
.external-link:hover {
background: #ff6b6b;
color: white;
}
.comments-section {
margin-top: 2rem;
padding-top: 2rem;
border-top: 2px solid #e5e7eb;
}
.comment-form {
margin-bottom: 1.5rem;
}
.comment-form textarea {
width: 100%;
padding: 0.75rem;
border: 1px solid #d1d5db;
border-radius: 4px;
font-size: 1rem;
resize: vertical;
margin-bottom: 0.75rem;
font-family: inherit;
}
.comment-form textarea:focus {
outline: none;
border-color: #ff6b6b;
}
.banned-notice {
background: #fef2f2;
border: 1px solid #fecaca;
color: #991b1b;
padding: 1rem;
border-radius: 4px;
text-align: center;
margin-bottom: 1.5rem;
font-size: 0.95rem;
}
.auth-prompt {
background: #f3f4f6;
padding: 1rem;
border-radius: 4px;
text-align: center;
margin-bottom: 1.5rem;
}
.auth-prompt p {
margin: 0;
color: #4b5563;
}
.comments-loading {
text-align: center;
padding: 1.5rem;
color: #6b7280;
font-size: 0.95rem;
}
.btn {
padding: 0.625rem 1.25rem;
border: none;
border-radius: 4px;
cursor: pointer;
font-size: 1rem;
transition: all 0.3s;
font-weight: 500;
}
.btn:hover {
opacity: 0.9;
}
.btn-primary {
background: #ff6b6b;
color: white;
}
@media (max-width: 640px) {
.container {
padding: 1rem;
}
.game-header {
flex-direction: column;
align-items: flex-start;
}
.game-header h1 {
font-size: 1.5rem;
}
.info-row {
flex-direction: column;
align-items: flex-start;
gap: 0.25rem;
}
.info-label {
min-width: auto;
}
}
`]
})
export class GameDetailComponent implements OnInit {
private readonly gamesService = inject(GamesService);
private readonly commentsService = inject(CommentsService);
readonly authService = inject(AuthService);
private readonly sanitizeService = inject(SanitizeService);
private readonly route = inject(ActivatedRoute);
private readonly router = inject(Router);
game = signal<Game | null>(null);
comments = signal<Comment[]>([]);
loading = signal(true);
commentsLoading = signal(false);
error = signal<string | null>(null);
newCommentContent = '';
showEditForm = signal(false);
ngOnInit() {
const gameId = this.route.snapshot.paramMap.get('id');
if (!gameId) {
this.error.set('No game ID provided');
this.loading.set(false);
return;
}
this.loadGame(gameId);
this.loadComments(gameId);
}
private loadGame(gameId: string) {
this.loading.set(true);
this.gamesService.getGameById(gameId).subscribe({
next: (game) => {
if (!game) {
this.error.set('Game not found');
} else {
this.game.set(game);
}
this.loading.set(false);
},
error: () => {
this.error.set('Failed to load game. It may not exist or there was an error.');
this.loading.set(false);
}
});
}
private loadComments(gameId: string) {
this.commentsLoading.set(true);
this.commentsService.getCommentsForGame(gameId).subscribe({
next: (comments) => {
this.comments.set(comments);
this.commentsLoading.set(false);
},
error: () => {
this.commentsLoading.set(false);
}
});
}
addComment() {
const game = this.game();
if (!game || !this.newCommentContent.trim()) return;
this.commentsService.addCommentToGame(game.id, { content: this.newCommentContent }).subscribe({
next: (comment) => {
this.comments.set([comment, ...this.comments()]);
this.newCommentContent = '';
}
});
}
handleCommentEdit(event: { commentId: string; content: string }) {
const game = this.game();
if (!game) return;
this.commentsService.updateCommentOnGame(game.id, event.commentId, event.content).subscribe({
next: (updatedComment) => {
this.comments.set(
this.comments().map(c => c.id === event.commentId ? updatedComment : c)
);
}
});
}
deleteComment(commentId: string) {
const game = this.game();
if (!game || !confirm('Are you sure you want to delete this comment?')) return;
this.commentsService.deleteCommentFromGame(game.id, commentId).subscribe({
next: () => {
this.comments.set(this.comments().filter(c => c.id !== commentId));
}
});
}
getStatusLabel(status: GameStatus): string {
switch (status) {
case GameStatus.playing: return 'Currently Playing';
case GameStatus.completed: return 'Completed';
case GameStatus.backlog: return 'In Backlog';
case GameStatus.retired: return 'Retired';
}
}
formatDate(date: Date | string): string {
return new Date(date).toLocaleDateString('en-GB', {
year: 'numeric',
month: 'long',
day: 'numeric'
});
}
formatTimeSpent(minutes: number): string {
const hours = Math.floor(minutes / 60);
const mins = minutes % 60;
if (hours === 0) {
return `${mins} minutes`;
} else if (mins === 0) {
return `${hours} hour${hours === 1 ? '' : 's'}`;
} else {
return `${hours} hour${hours === 1 ? '' : 's'} ${mins} minute${mins === 1 ? '' : 's'}`;
}
}
toggleEditForm() {
this.showEditForm.update(value => !value);
}
saveEdit(data: UpdateGameDto) {
const game = this.game();
if (!game) return;
this.gamesService.updateGame(game.id, data).subscribe({
next: (updatedGame) => {
this.game.set(updatedGame);
this.showEditForm.set(false);
},
error: (err) => {
alert('Failed to update game: ' + (err.error?.message || 'Unknown error'));
}
});
}
cancelEdit() {
this.showEditForm.set(false);
}
deleteGame() {
const game = this.game();
if (!game) return;
if (!confirm(`Are you sure you want to delete "${game.title}"? This action cannot be undone.`)) {
return;
}
this.gamesService.deleteGame(game.id).subscribe({
next: () => {
this.router.navigate(['/games']);
},
error: (err) => {
alert('Failed to delete game: ' + (err.error?.message || 'Unknown error'));
}
});
}
}
File diff suppressed because it is too large Load Diff
@@ -6,7 +6,7 @@
import { Component, inject, signal, OnInit } from '@angular/core';
import { CommonModule } from '@angular/common';
import { RouterModule, Router } from '@angular/router';
import { RouterModule } from '@angular/router';
import { AuthService } from '../../services/auth.service';
import { ApiService } from '../../services/api.service';
@@ -16,71 +16,36 @@ import { ApiService } from '../../services/api.service';
imports: [CommonModule, RouterModule],
template: `
<header class="header">
<nav class="navbar" aria-label="Main navigation">
<nav class="navbar">
<div class="nav-brand">
<img src="/assets/nav-icon.jpg" alt="" class="brand-icon" role="presentation" />
<h1><a routerLink="/">Naomi's Library</a></h1>
@if (version()) {
<span class="version" aria-label="Version {{ version() }}">v{{ version() }}</span>
<span class="version">v{{ version() }}</span>
}
</div>
<ul class="nav-links" role="list">
<li><a routerLink="/games" routerLinkActive="active" [attr.aria-current]="isCurrentRoute('/games') ? 'page' : null">Games</a></li>
<li><a routerLink="/books" routerLinkActive="active" [attr.aria-current]="isCurrentRoute('/books') ? 'page' : null">Books</a></li>
<li><a routerLink="/music" routerLinkActive="active" [attr.aria-current]="isCurrentRoute('/music') ? 'page' : null">Music</a></li>
<li><a routerLink="/shows" routerLinkActive="active" [attr.aria-current]="isCurrentRoute('/shows') ? 'page' : null">Shows</a></li>
<li><a routerLink="/manga" routerLinkActive="active" [attr.aria-current]="isCurrentRoute('/manga') ? 'page' : null">Manga</a></li>
<li><a routerLink="/art" routerLinkActive="active" [attr.aria-current]="isCurrentRoute('/art') ? 'page' : null">Art</a></li>
<ul class="nav-links">
<li><a routerLink="/games" routerLinkActive="active">Games</a></li>
<li><a routerLink="/books" routerLinkActive="active">Books</a></li>
<li><a routerLink="/music" routerLinkActive="active">Music</a></li>
<li><a routerLink="/shows" routerLinkActive="active">Shows</a></li>
<li><a routerLink="/manga" routerLinkActive="active">Manga</a></li>
<li><a routerLink="/art" routerLinkActive="active">Art</a></li>
</ul>
<div class="auth-section">
@if (authService.user(); as user) {
<div class="user-menu">
@if (user.avatar) {
<button
class="user-avatar-button"
[attr.aria-label]="'User menu for ' + user.username"
[attr.aria-expanded]="showDropdown()"
aria-haspopup="true"
(click)="toggleDropdown()"
(keydown.escape)="closeDropdown()"
>
<img
[src]="user.avatar"
[alt]="'Avatar for ' + user.username"
class="user-avatar"
/>
</button>
}
@if (showDropdown()) {
<div
class="dropdown-menu"
role="menu"
aria-label="User menu"
tabindex="-1"
(keydown.escape)="closeDropdown()"
>
<a [routerLink]="['/profile', user.slug || user.id]" class="dropdown-item" role="menuitem" (click)="closeDropdown()">My Profile</a>
<a routerLink="/settings" class="dropdown-item" role="menuitem" (click)="closeDropdown()">Settings</a>
<a routerLink="/achievements" class="dropdown-item" role="menuitem" (click)="closeDropdown()"><span aria-hidden="true">🏆</span> Achievements</a>
<a routerLink="/leaderboard" class="dropdown-item" role="menuitem" (click)="closeDropdown()"><span aria-hidden="true">🏆</span> Leaderboard</a>
<a routerLink="/activity" class="dropdown-item" role="menuitem" (click)="closeDropdown()"><span aria-hidden="true">📰</span> Activity Feed</a>
<a routerLink="/about" class="dropdown-item" role="menuitem" (click)="closeDropdown()"><span aria-hidden="true">️</span> About</a>
@if (!user.isAdmin) {
<a routerLink="/my-suggestions" class="dropdown-item" role="menuitem" (click)="closeDropdown()">My Suggestions</a>
}
<a routerLink="/my-likes" class="dropdown-item" role="menuitem" (click)="closeDropdown()">My Likes</a>
@if (user.isAdmin) {
<a routerLink="/admin/users" class="dropdown-item" role="menuitem" (click)="closeDropdown()">Users</a>
<a routerLink="/admin/audit" class="dropdown-item" role="menuitem" (click)="closeDropdown()">Audit</a>
<a routerLink="/admin/suggestions" class="dropdown-item" role="menuitem" (click)="closeDropdown()">Suggestions</a>
<a routerLink="/admin/reports" class="dropdown-item" role="menuitem" (click)="closeDropdown()">Reports</a>
}
<button (click)="logout()" class="dropdown-item logout-btn" role="menuitem">Logout</button>
</div>
}
</div>
<span class="welcome">Welcome, {{ user.username }}!</span>
@if (!user.isAdmin) {
<a routerLink="/my-suggestions" class="user-link">My Suggestions</a>
}
<a routerLink="/my-likes" class="user-link">My Likes</a>
@if (user.isAdmin) {
<a routerLink="/admin/users" class="admin-badge">Users</a>
<a routerLink="/admin/audit" class="admin-badge">Audit</a>
<a routerLink="/admin/suggestions" class="admin-badge">Suggestions</a>
}
<button (click)="logout()" class="btn btn-secondary">Logout</button>
} @else {
<button (click)="login()" class="btn btn-primary">Login with Discord</button>
}
@@ -105,27 +70,6 @@ import { ApiService } from '../../services/api.service';
margin: 0 auto;
}
.nav-brand {
display: flex;
align-items: center;
gap: 0.75rem;
}
.brand-icon {
width: 40px;
height: 40px;
border-radius: 50%;
object-fit: cover;
border: 2px solid var(--witch-purple);
box-shadow: 0 2px 8px rgba(157, 78, 221, 0.3);
transition: transform 0.2s, box-shadow 0.2s;
}
.brand-icon:hover {
transform: scale(1.05);
box-shadow: 0 4px 12px rgba(157, 78, 221, 0.5);
}
.nav-brand h1 {
margin: 0;
font-size: 1.5rem;
@@ -178,90 +122,6 @@ import { ApiService } from '../../services/api.service';
color: var(--witch-lavender);
}
.user-menu {
position: relative;
}
.user-avatar-button {
background: none;
border: none;
padding: 0;
cursor: pointer;
display: flex;
align-items: center;
}
.user-avatar {
width: 40px;
height: 40px;
border-radius: 50%;
border: 2px solid var(--witch-lavender);
transition: all 0.3s;
}
.user-avatar-button:hover .user-avatar,
.user-avatar-button:focus .user-avatar {
border-color: var(--witch-moon);
transform: scale(1.1);
}
.user-avatar-button:focus-visible {
outline: 3px solid var(--witch-rose);
outline-offset: 2px;
border-radius: 50%;
}
.dropdown-menu {
position: absolute;
top: 50px;
right: 0;
background-color: var(--witch-purple);
border: 2px solid var(--witch-lavender);
border-radius: 8px;
padding: 0.5rem 0;
min-width: 180px;
box-shadow: 0 4px 12px var(--witch-shadow);
z-index: 1000;
animation: fadeIn 0.2s ease-in;
}
@keyframes fadeIn {
from {
opacity: 0;
transform: translateY(-10px);
}
to {
opacity: 1;
transform: translateY(0);
}
}
.dropdown-item {
display: block;
width: 100%;
padding: 0.75rem 1rem;
color: var(--witch-lavender);
text-decoration: none;
background: none;
border: none;
text-align: left;
font-size: 0.9rem;
cursor: pointer;
transition: all 0.2s;
}
.dropdown-item:hover {
background-color: var(--witch-plum);
color: var(--witch-moon);
}
.logout-btn {
border-top: 1px solid var(--witch-lavender);
margin-top: 0.5rem;
padding-top: 0.75rem;
font-weight: 500;
}
.admin-badge {
background-color: var(--witch-rose);
color: var(--witch-moon);
@@ -329,9 +189,7 @@ import { ApiService } from '../../services/api.service';
export class HeaderComponent implements OnInit {
authService = inject(AuthService);
private apiService = inject(ApiService);
private router = inject(Router);
version = signal<string | null>(null);
showDropdown = signal<boolean>(false);
ngOnInit() {
this.apiService.get<{ version: string }>('/version').subscribe({
@@ -340,24 +198,11 @@ export class HeaderComponent implements OnInit {
});
}
isCurrentRoute(route: string): boolean {
return this.router.url.startsWith(route);
}
toggleDropdown() {
this.showDropdown.update(v => !v);
}
closeDropdown() {
this.showDropdown.set(false);
}
login() {
this.authService.login();
}
logout() {
this.closeDropdown();
this.authService.logout().subscribe();
}
}
@@ -23,7 +23,6 @@ import { Game, GameStatus, Book, BookStatus, Music, MusicType, Manga, MangaStatu
<div class="container">
<div class="hero">
<h1>Welcome to Naomi's Library</h1>
<img src="/assets/nav-icon.jpg" alt="Naomi's avatar" class="hero-avatar" />
<p class="tagline">A personal collection of games, books, music, manga, shows, and art</p>
</div>
@@ -93,7 +92,7 @@ import { Game, GameStatus, Book, BookStatus, Music, MusicType, Manga, MangaStatu
<ul class="recent-list">
@for (game of recentGames(); track game.id) {
<li>
<a [routerLink]="['/games', game.id]">{{ game.title }}</a>
<a routerLink="/games">{{ game.title }}</a>
@if (game.platform) {
<span class="platform">({{ game.platform }})</span>
}
@@ -109,7 +108,7 @@ import { Game, GameStatus, Book, BookStatus, Music, MusicType, Manga, MangaStatu
<ul class="recent-list">
@for (book of recentBooks(); track book.id) {
<li>
<a [routerLink]="['/books', book.id]">{{ book.title }}</a>
<a routerLink="/books">{{ book.title }}</a>
<span class="author">by {{ book.author }}</span>
</li>
}
@@ -123,7 +122,7 @@ import { Game, GameStatus, Book, BookStatus, Music, MusicType, Manga, MangaStatu
<ul class="recent-list">
@for (music of recentMusic(); track music.id) {
<li>
<a [routerLink]="['/music', music.id]">{{ music.title }}</a>
<a routerLink="/music">{{ music.title }}</a>
<span class="artist">by {{ music.artist }}</span>
</li>
}
@@ -137,7 +136,7 @@ import { Game, GameStatus, Book, BookStatus, Music, MusicType, Manga, MangaStatu
<ul class="recent-list">
@for (manga of recentManga(); track manga.id) {
<li>
<a [routerLink]="['/manga', manga.id]">{{ manga.title }}</a>
<a routerLink="/manga">{{ manga.title }}</a>
<span class="author">by {{ manga.author }}</span>
</li>
}
@@ -151,7 +150,7 @@ import { Game, GameStatus, Book, BookStatus, Music, MusicType, Manga, MangaStatu
<ul class="recent-list">
@for (show of recentShows(); track show.id) {
<li>
<a [routerLink]="['/shows', show.id]">{{ show.title }}</a>
<a routerLink="/shows">{{ show.title }}</a>
<span class="show-type">{{ formatShowType(show.type) }}</span>
</li>
}
@@ -165,7 +164,7 @@ import { Game, GameStatus, Book, BookStatus, Music, MusicType, Manga, MangaStatu
<ul class="recent-list">
@for (art of recentArt(); track art.id) {
<li>
<a [routerLink]="['/art', art.id]">{{ art.title }}</a>
<a routerLink="/art">{{ art.title }}</a>
<span class="artist">by {{ art.artist }}</span>
</li>
}
@@ -191,28 +190,10 @@ import { Game, GameStatus, Book, BookStatus, Music, MusicType, Manga, MangaStatu
.hero h1 {
font-size: 2.5rem;
margin-bottom: 1rem;
margin-bottom: 0.5rem;
color: var(--witch-purple);
}
.hero-avatar {
width: 150px;
height: 150px;
border-radius: 50%;
object-fit: cover;
border: 4px solid var(--witch-lavender);
box-shadow: 0 4px 12px var(--witch-shadow);
margin: 1rem auto;
display: block;
transition: all 0.3s;
}
.hero-avatar:hover {
transform: scale(1.05);
box-shadow: 0 8px 16px rgba(157, 78, 221, 0.5);
border-color: var(--witch-rose);
}
.tagline {
font-size: 1.2rem;
color: var(--witch-plum);
@@ -1,545 +0,0 @@
/**
* @copyright 2026 NHCarrigan
* @license Naomi's Public License
* @author Naomi Carrigan
*/
import { Component, OnInit, inject, signal } from '@angular/core';
import { CommonModule } from '@angular/common';
import { RouterModule } from '@angular/router';
import type {
SuggestionsLeaderboard,
LikesLeaderboard,
CommentsLeaderboard,
OverallLeaderboard,
} from '@library/shared-types';
import { LeaderboardService } from '../../services/leaderboard.service';
import { AuthService } from '../../services/auth.service';
type LeaderboardTab = 'overall' | 'suggestions' | 'likes' | 'comments';
@Component({
selector: 'app-leaderboard',
standalone: true,
imports: [CommonModule, RouterModule],
template: `
<div class="container">
<h1>🏆 Community Leaderboard</h1>
<p class="subtitle">Celebrating our most engaged community members!</p>
<div class="tabs">
<button
(click)="setTab('overall')"
[class.active]="activeTab() === 'overall'"
class="tab-btn"
>
🌟 Overall
</button>
<button
(click)="setTab('suggestions')"
[class.active]="activeTab() === 'suggestions'"
class="tab-btn"
>
💡 Suggestions
</button>
<button
(click)="setTab('likes')"
[class.active]="activeTab() === 'likes'"
class="tab-btn"
>
❤️ Likes
</button>
<button
(click)="setTab('comments')"
[class.active]="activeTab() === 'comments'"
class="tab-btn"
>
💬 Comments
</button>
</div>
@if (loading()) {
<div class="loading">Loading leaderboard...</div>
} @else {
@if (activeTab() === 'overall') {
<div class="leaderboard">
<h2>Overall Leaders</h2>
<p class="description">Ranked by achievement points, diversity of engagement, and total activity</p>
@if (overallLeaderboard().length === 0) {
<p class="empty">No users on the leaderboard yet!</p>
} @else {
<div class="leaderboard-list">
@for (user of overallLeaderboard(); track user.id; let i = $index) {
<div class="leaderboard-item" [class.highlight]="user.id === authService.user()?.id">
<div class="rank">
@if (i === 0) {
<span class="medal">🥇</span>
} @else if (i === 1) {
<span class="medal">🥈</span>
} @else if (i === 2) {
<span class="medal">🥉</span>
} @else {
<span class="rank-number">#{{ i + 1 }}</span>
}
</div>
<div class="user-info">
<div class="user-header">
@if (user.avatar) {
<img [src]="user.avatar" [alt]="user.username" class="avatar">
}
<div class="user-details">
<a [routerLink]="['/profile', user.slug || user.id]" class="username">
{{ user.username }}
</a>
<div class="badges">
@if (user.primaryBadge === 'STAFF') {
<span class="badge staff-badge">STAFF</span>
}
@if (user.primaryBadge === 'MOD') {
<span class="badge mod-badge">MOD</span>
}
@if (user.primaryBadge === 'VIP') {
<span class="badge vip-badge">VIP</span>
}
</div>
</div>
</div>
<div class="stats">
<span class="stat">🎯 {{ user.achievementPoints }} pts</span>
<span class="stat">🏅 {{ user.achievementCount }} achievements</span>
<span class="stat">💡 {{ user.totalSuggestions }} suggestions</span>
<span class="stat">❤️ {{ user.totalLikes }} likes</span>
<span class="stat">💬 {{ user.totalComments }} comments</span>
<span class="stat">🔥 {{ user.currentStreak }} day streak</span>
</div>
</div>
</div>
}
</div>
}
</div>
}
@if (activeTab() === 'suggestions') {
<div class="leaderboard">
<h2>Top Suggestions</h2>
<p class="description">Ranked by total suggestions and acceptance rate</p>
@if (suggestionsLeaderboard().length === 0) {
<p class="empty">No users on the leaderboard yet!</p>
} @else {
<div class="leaderboard-list">
@for (user of suggestionsLeaderboard(); track user.id; let i = $index) {
<div class="leaderboard-item" [class.highlight]="user.id === authService.user()?.id">
<div class="rank">
@if (i === 0) {
<span class="medal">🥇</span>
} @else if (i === 1) {
<span class="medal">🥈</span>
} @else if (i === 2) {
<span class="medal">🥉</span>
} @else {
<span class="rank-number">#{{ i + 1 }}</span>
}
</div>
<div class="user-info">
<div class="user-header">
@if (user.avatar) {
<img [src]="user.avatar" [alt]="user.username" class="avatar">
}
<div class="user-details">
<a [routerLink]="['/profile', user.slug || user.id]" class="username">
{{ user.username }}
</a>
<div class="badges">
@if (user.primaryBadge === 'STAFF') {
<span class="badge staff-badge">STAFF</span>
}
@if (user.primaryBadge === 'MOD') {
<span class="badge mod-badge">MOD</span>
}
@if (user.primaryBadge === 'VIP') {
<span class="badge vip-badge">VIP</span>
}
</div>
</div>
</div>
<div class="stats">
<span class="stat">💡 {{ user.totalSuggestions }} suggestions</span>
<span class="stat">✅ {{ user.acceptedSuggestions }} accepted</span>
<span class="stat">📊 {{ user.acceptanceRate }}% acceptance rate</span>
</div>
</div>
</div>
}
</div>
}
</div>
}
@if (activeTab() === 'likes') {
<div class="leaderboard">
<h2>Top Likers</h2>
<p class="description">Ranked by total likes given</p>
@if (likesLeaderboard().length === 0) {
<p class="empty">No users on the leaderboard yet!</p>
} @else {
<div class="leaderboard-list">
@for (user of likesLeaderboard(); track user.id; let i = $index) {
<div class="leaderboard-item" [class.highlight]="user.id === authService.user()?.id">
<div class="rank">
@if (i === 0) {
<span class="medal">🥇</span>
} @else if (i === 1) {
<span class="medal">🥈</span>
} @else if (i === 2) {
<span class="medal">🥉</span>
} @else {
<span class="rank-number">#{{ i + 1 }}</span>
}
</div>
<div class="user-info">
<div class="user-header">
@if (user.avatar) {
<img [src]="user.avatar" [alt]="user.username" class="avatar">
}
<div class="user-details">
<a [routerLink]="['/profile', user.slug || user.id]" class="username">
{{ user.username }}
</a>
<div class="badges">
@if (user.primaryBadge === 'STAFF') {
<span class="badge staff-badge">STAFF</span>
}
@if (user.primaryBadge === 'MOD') {
<span class="badge mod-badge">MOD</span>
}
@if (user.primaryBadge === 'VIP') {
<span class="badge vip-badge">VIP</span>
}
</div>
</div>
</div>
<div class="stats">
<span class="stat">❤️ {{ user.totalLikes }} likes given</span>
</div>
</div>
</div>
}
</div>
}
</div>
}
@if (activeTab() === 'comments') {
<div class="leaderboard">
<h2>Top Commenters</h2>
<p class="description">Ranked by total comments posted</p>
@if (commentsLeaderboard().length === 0) {
<p class="empty">No users on the leaderboard yet!</p>
} @else {
<div class="leaderboard-list">
@for (user of commentsLeaderboard(); track user.id; let i = $index) {
<div class="leaderboard-item" [class.highlight]="user.id === authService.user()?.id">
<div class="rank">
@if (i === 0) {
<span class="medal">🥇</span>
} @else if (i === 1) {
<span class="medal">🥈</span>
} @else if (i === 2) {
<span class="medal">🥉</span>
} @else {
<span class="rank-number">#{{ i + 1 }}</span>
}
</div>
<div class="user-info">
<div class="user-header">
@if (user.avatar) {
<img [src]="user.avatar" [alt]="user.username" class="avatar">
}
<div class="user-details">
<a [routerLink]="['/profile', user.slug || user.id]" class="username">
{{ user.username }}
</a>
<div class="badges">
@if (user.primaryBadge === 'STAFF') {
<span class="badge staff-badge">STAFF</span>
}
@if (user.primaryBadge === 'MOD') {
<span class="badge mod-badge">MOD</span>
}
@if (user.primaryBadge === 'VIP') {
<span class="badge vip-badge">VIP</span>
}
</div>
</div>
</div>
<div class="stats">
<span class="stat">💬 {{ user.totalComments }} comments</span>
</div>
</div>
</div>
}
</div>
}
</div>
}
}
</div>
`,
styles: [`
.container {
max-width: 1200px;
margin: 0 auto;
padding: 2rem;
}
h1 {
text-align: center;
color: var(--witch-purple);
margin-bottom: 0.5rem;
}
.subtitle {
text-align: center;
color: var(--witch-plum);
margin-bottom: 2rem;
font-size: 1.1rem;
}
.tabs {
display: flex;
gap: 1rem;
justify-content: center;
margin-bottom: 2rem;
flex-wrap: wrap;
}
.tab-btn {
padding: 0.75rem 1.5rem;
background: var(--witch-lavender);
color: var(--witch-purple);
border: 2px solid var(--witch-lavender);
border-radius: 8px;
cursor: pointer;
font-size: 1rem;
font-weight: 600;
transition: all 0.3s;
}
.tab-btn:hover {
background: var(--witch-mauve);
transform: translateY(-2px);
box-shadow: 0 4px 8px var(--witch-shadow);
}
.tab-btn.active {
background: var(--witch-rose);
color: var(--witch-moon);
border-color: var(--witch-rose);
}
.loading {
text-align: center;
padding: 3rem;
color: var(--witch-plum);
font-size: 1.2rem;
}
.leaderboard h2 {
color: var(--witch-purple);
margin-bottom: 0.5rem;
}
.description {
color: var(--witch-plum);
margin-bottom: 1.5rem;
font-size: 0.9rem;
}
.empty {
text-align: center;
padding: 3rem;
color: var(--witch-mauve);
font-style: italic;
}
.leaderboard-list {
display: flex;
flex-direction: column;
gap: 1rem;
}
.leaderboard-item {
display: flex;
align-items: center;
gap: 1rem;
background: rgba(255, 255, 255, 0.95);
border: 2px solid var(--witch-lavender);
border-radius: 8px;
padding: 1rem;
transition: all 0.3s;
}
.leaderboard-item:hover {
transform: translateX(4px);
box-shadow: 0 4px 12px var(--witch-shadow);
border-color: var(--witch-mauve);
}
.leaderboard-item.highlight {
background: linear-gradient(135deg, rgba(255, 215, 245, 0.3), rgba(255, 240, 250, 0.3));
border-color: var(--witch-rose);
box-shadow: 0 0 20px rgba(168, 87, 126, 0.2);
}
.rank {
min-width: 60px;
text-align: center;
}
.medal {
font-size: 2rem;
}
.rank-number {
font-size: 1.5rem;
font-weight: 700;
color: var(--witch-plum);
}
.user-info {
flex: 1;
}
.user-header {
display: flex;
align-items: center;
gap: 1rem;
margin-bottom: 0.75rem;
}
.avatar {
width: 48px;
height: 48px;
border-radius: 50%;
border: 2px solid var(--witch-lavender);
}
.user-details {
flex: 1;
}
.username {
font-size: 1.1rem;
font-weight: 600;
color: var(--witch-purple);
text-decoration: none;
transition: color 0.2s;
}
.username:hover {
color: var(--witch-rose);
}
.badges {
display: flex;
gap: 0.5rem;
margin-top: 0.25rem;
}
.badge {
padding: 0.15rem 0.5rem;
border-radius: 4px;
font-size: 0.7rem;
font-weight: 600;
}
.staff-badge {
background: linear-gradient(135deg, #e84393, #fd79a8);
color: white;
}
.mod-badge {
background: linear-gradient(135deg, #00b894, #00cec9);
color: white;
}
.vip-badge {
background: linear-gradient(135deg, #ffd700, #ffaa00);
color: #1a1a1a;
}
.stats {
display: flex;
flex-wrap: wrap;
gap: 1rem;
}
.stat {
font-size: 0.9rem;
color: var(--witch-plum);
font-weight: 500;
}
@media (max-width: 768px) {
.tabs {
gap: 0.5rem;
}
.tab-btn {
padding: 0.5rem 1rem;
font-size: 0.9rem;
}
.leaderboard-item {
flex-direction: column;
align-items: flex-start;
}
.rank {
min-width: auto;
}
.stats {
flex-direction: column;
gap: 0.5rem;
}
}
`]
})
export class LeaderboardComponent implements OnInit {
leaderboardService = inject(LeaderboardService);
authService = inject(AuthService);
activeTab = signal<LeaderboardTab>('overall');
loading = signal(true);
overallLeaderboard = signal<OverallLeaderboard[]>([]);
suggestionsLeaderboard = signal<SuggestionsLeaderboard[]>([]);
likesLeaderboard = signal<LikesLeaderboard[]>([]);
commentsLeaderboard = signal<CommentsLeaderboard[]>([]);
ngOnInit() {
this.loadLeaderboards();
}
loadLeaderboards() {
this.loading.set(true);
this.leaderboardService.getAllLeaderboards(25).subscribe({
next: (data) => {
this.overallLeaderboard.set(data.topOverall);
this.suggestionsLeaderboard.set(data.topSuggestions);
this.likesLeaderboard.set(data.topLikes);
this.commentsLeaderboard.set(data.topComments);
this.loading.set(false);
},
error: () => {
this.loading.set(false);
}
});
}
setTab(tab: LeaderboardTab) {
this.activeTab.set(tab);
}
}
@@ -1,747 +0,0 @@
/**
* @copyright 2026 NHCarrigan
* @license Naomi's Public License
* @author Naomi Carrigan
*/
import { Component, OnInit, inject, signal } from '@angular/core';
import { CommonModule } from '@angular/common';
import { RouterLink, ActivatedRoute, Router } from '@angular/router';
import { FormsModule } from '@angular/forms';
import { MangaService } from '../../services/manga.service';
import { CommentsService } from '../../services/comments.service';
import { AuthService } from '../../services/auth.service';
import { SanitizeService } from '../../services/sanitize.service';
import { CommentDisplayComponent } from '../comment-display/comment-display.component';
import { LikeButtonComponent } from '../shared/like-button.component';
import { MangaFormComponent } from '../shared/manga-form.component';
import { Manga, Comment, MangaStatus, UpdateMangaDto } from '@library/shared-types';
@Component({
selector: 'app-manga-detail',
standalone: true,
imports: [CommonModule, RouterLink, FormsModule, CommentDisplayComponent, LikeButtonComponent, MangaFormComponent],
template: `
<div class="container">
<div class="breadcrumb">
<a routerLink="/manga" class="breadcrumb-link">← Back to Manga</a>
</div>
@if (showEditForm() && authService.user()?.isAdmin && manga()) {
<app-manga-form
mode="edit"
[manga]="manga()!"
(formSubmit)="saveEdit($event)"
(formCancel)="cancelEdit()"
></app-manga-form>
}
@if (loading()) {
<div class="loading">Loading manga details...</div>
} @else if (error()) {
<div class="error-state">
<h2>Manga Not Found</h2>
<p>{{ error() }}</p>
<a routerLink="/manga" class="btn btn-primary">Return to Manga</a>
</div>
} @else if (manga()) {
<div class="manga-detail-card">
<div class="manga-cover-section">
<img [src]="manga()!.coverImage || '/assets/default-cover.jpg'" [alt]="manga()!.title" class="manga-cover-large">
</div>
<div class="manga-content">
<div class="manga-header">
<h1>{{ manga()!.title }}</h1>
<span class="status status-{{ manga()!.status }}">
{{ getStatusLabel(manga()!.status) }}
</span>
</div>
<p class="author">by {{ manga()!.author }}</p>
@if (authService.user()?.isAdmin) {
<div class="admin-actions">
<button (click)="toggleEditForm()" class="btn btn-edit">✏️ {{ showEditForm() ? 'Cancel Edit' : 'Edit' }}</button>
<button (click)="deleteManga()" class="btn btn-delete">🗑️ Delete</button>
</div>
}
@if (manga()!.rating) {
<div class="info-row">
<span class="info-label">Rating:</span>
<div class="rating">
@for (star of [1,2,3,4,5,6,7,8,9,10]; track star) {
<span [class.filled]="star <= manga()!.rating!">★</span>
}
<span class="rating-text">({{ manga()!.rating }}/10)</span>
</div>
</div>
}
@if (manga()!.timeSpent) {
<div class="info-row">
<span class="info-label">Reading Time:</span>
<span class="info-value time-spent">{{ formatTimeSpent(manga()!.timeSpent!) }}</span>
</div>
}
@if (manga()!.dateStarted) {
<div class="info-row">
<span class="info-label">Started:</span>
<span class="info-value">{{ formatDate(manga()!.dateStarted!) }}</span>
</div>
}
@if (manga()!.dateFinished) {
<div class="info-row">
<span class="info-label">Finished:</span>
<span class="info-value">{{ formatDate(manga()!.dateFinished!) }}</span>
</div>
}
<div class="info-row">
<span class="info-label">Added:</span>
<span class="info-value">{{ formatDate(manga()!.createdAt) }}</span>
</div>
<div class="info-row">
<span class="info-label">Last Updated:</span>
<span class="info-value">{{ formatDate(manga()!.updatedAt) }}</span>
</div>
<div class="like-section">
<app-like-button
entityType="manga"
[entityId]="manga()!.id"
></app-like-button>
</div>
@if (manga()!.notes) {
<div class="notes-section">
<h3>Notes</h3>
<p class="notes">{{ manga()!.notes }}</p>
</div>
}
@if (manga()!.tags && manga()!.tags.length > 0) {
<div class="tags-section">
<h3>Tags</h3>
<div class="tags-display">
@for (tag of manga()!.tags; track tag) {
<span class="tag-chip">{{ tag }}</span>
}
</div>
</div>
}
@if (manga()!.links && manga()!.links.length > 0) {
<div class="links-section">
<h3>External Links</h3>
<div class="links-display">
@for (link of manga()!.links; track link.url) {
<a [href]="link.url" target="_blank" rel="noopener noreferrer" class="external-link">
{{ link.title }} ↗
</a>
}
</div>
</div>
}
<div class="comments-section">
<h3>Comments</h3>
@if (authService.isAuthenticated()) {
@if (authService.user()?.isBanned) {
<div class="banned-notice">
You have been banned from commenting.
</div>
} @else {
<form (ngSubmit)="addComment()" class="comment-form">
<textarea
[(ngModel)]="newCommentContent"
name="comment"
placeholder="Add a comment (Markdown supported)..."
rows="3"
></textarea>
<button type="submit" class="btn btn-primary">Post Comment</button>
</form>
}
} @else {
<div class="auth-prompt">
<p>Please sign in to comment.</p>
</div>
}
@if (commentsLoading()) {
<div class="comments-loading">Loading comments...</div>
} @else {
<app-comment-display
[comments]="comments"
(edit)="handleCommentEdit($event)"
(delete)="deleteComment($event)"
/>
}
</div>
</div>
</div>
}
</div>
`,
styles: [`
.container {
max-width: 900px;
margin: 0 auto;
padding: 2rem;
}
.breadcrumb {
margin-bottom: 1.5rem;
}
.breadcrumb-link {
color: #f59e0b;
text-decoration: none;
font-weight: 500;
transition: opacity 0.3s;
}
.breadcrumb-link:hover {
opacity: 0.8;
}
.loading {
text-align: center;
padding: 3rem;
color: #666;
font-size: 1.1rem;
}
.error-state {
text-align: center;
padding: 3rem;
background: #fef2f2;
border: 1px solid #fecaca;
border-radius: 8px;
}
.error-state h2 {
color: #991b1b;
margin-bottom: 1rem;
}
.error-state p {
color: #dc2626;
margin-bottom: 1.5rem;
}
.manga-detail-card {
background: white;
border: 1px solid #e5e7eb;
border-radius: 8px;
overflow: hidden;
box-shadow: 0 1px 3px rgba(0,0,0,0.1);
}
.manga-cover-section {
width: 100%;
background: #f3f4f6;
display: flex;
justify-content: center;
align-items: center;
padding: 2rem;
}
.manga-cover-large {
max-width: 100%;
max-height: 400px;
object-fit: contain;
border-radius: 4px;
}
.manga-content {
padding: 2rem;
}
.manga-header {
display: flex;
align-items: center;
gap: 1rem;
margin-bottom: 1rem;
flex-wrap: wrap;
}
.manga-header h1 {
margin: 0;
font-size: 2rem;
color: #1f2937;
}
.author {
color: #6b7280;
font-style: italic;
font-size: 1.1rem;
margin: 0 0 1rem 0;
}
.status {
display: inline-block;
padding: 0.5rem 1rem;
border-radius: 4px;
font-size: 0.9rem;
font-weight: 500;
}
.status-reading {
background: #fef3c7;
color: #92400e;
}
.status-completed {
background: #d1fae5;
color: #065f46;
}
.status-wantToRead {
background: #e0e7ff;
color: #3730a3;
}
.status-retired {
background: #f3f4f6;
color: #4b5563;
}
.admin-actions {
display: flex;
gap: 0.75rem;
margin-bottom: 1.5rem;
padding: 1rem;
background: #fef3c7;
border-radius: 4px;
border: 1px solid #fbbf24;
}
.btn-edit {
background: #3b82f6;
color: white;
}
.btn-edit:hover {
background: #2563eb;
}
.btn-delete {
background: #ef4444;
color: white;
}
.btn-delete:hover {
background: #dc2626;
}
.info-row {
display: flex;
align-items: center;
gap: 1rem;
margin-bottom: 0.75rem;
padding: 0.5rem 0;
border-bottom: 1px solid #f3f4f6;
}
.info-label {
font-weight: 600;
color: #4b5563;
min-width: 120px;
}
.info-value {
color: #1f2937;
}
.time-spent {
color: #f59e0b;
font-weight: 600;
}
.rating {
display: flex;
align-items: center;
gap: 0.25rem;
}
.rating span {
color: #e5e7eb;
font-size: 1.2rem;
}
.rating span.filled {
color: #f59e0b;
}
.rating-text {
margin-left: 0.5rem;
font-size: 1rem;
color: #6b7280;
font-weight: 500;
}
.like-section {
margin: 1.5rem 0;
}
.notes-section,
.tags-section,
.links-section {
margin-top: 2rem;
}
.notes-section h3,
.tags-section h3,
.links-section h3,
.comments-section h3 {
font-size: 1.25rem;
color: #1f2937;
margin-bottom: 1rem;
}
.notes {
font-size: 1rem;
color: #4b5563;
line-height: 1.6;
white-space: pre-wrap;
}
.tags-display {
display: flex;
flex-wrap: wrap;
gap: 0.5rem;
}
.tag-chip {
background: #f59e0b;
color: white;
padding: 0.375rem 0.875rem;
border-radius: 12px;
font-size: 0.875rem;
font-weight: 500;
}
.links-display {
display: flex;
flex-wrap: wrap;
gap: 0.75rem;
}
.external-link {
color: #f59e0b;
text-decoration: none;
font-size: 0.95rem;
padding: 0.5rem 1rem;
border: 2px solid #f59e0b;
border-radius: 4px;
transition: all 0.2s;
font-weight: 500;
}
.external-link:hover {
background: #f59e0b;
color: white;
}
.comments-section {
margin-top: 2rem;
padding-top: 2rem;
border-top: 2px solid #e5e7eb;
}
.comment-form {
margin-bottom: 1.5rem;
}
.comment-form textarea {
width: 100%;
padding: 0.75rem;
border: 1px solid #d1d5db;
border-radius: 4px;
font-size: 1rem;
resize: vertical;
margin-bottom: 0.75rem;
font-family: inherit;
}
.comment-form textarea:focus {
outline: none;
border-color: #f59e0b;
}
.banned-notice {
background: #fef2f2;
border: 1px solid #fecaca;
color: #991b1b;
padding: 1rem;
border-radius: 4px;
text-align: center;
margin-bottom: 1.5rem;
font-size: 0.95rem;
}
.auth-prompt {
background: #f3f4f6;
padding: 1rem;
border-radius: 4px;
text-align: center;
margin-bottom: 1.5rem;
}
.auth-prompt p {
margin: 0;
color: #4b5563;
}
.comments-loading {
text-align: center;
padding: 1.5rem;
color: #6b7280;
font-size: 0.95rem;
}
.btn {
padding: 0.625rem 1.25rem;
border: none;
border-radius: 4px;
cursor: pointer;
font-size: 1rem;
transition: all 0.3s;
font-weight: 500;
}
.btn:hover {
opacity: 0.9;
}
.btn-primary {
background: #f59e0b;
color: white;
}
@media (max-width: 640px) {
.container {
padding: 1rem;
}
.manga-header {
flex-direction: column;
align-items: flex-start;
}
.manga-header h1 {
font-size: 1.5rem;
}
.admin-actions {
display: flex;
gap: 0.75rem;
margin-bottom: 1.5rem;
padding: 1rem;
background: #fef3c7;
border-radius: 4px;
border: 1px solid #fbbf24;
}
.btn-edit {
background: #3b82f6;
color: white;
}
.btn-edit:hover {
background: #2563eb;
}
.btn-delete {
background: #ef4444;
color: white;
}
.btn-delete:hover {
background: #dc2626;
}
.info-row {
flex-direction: column;
align-items: flex-start;
gap: 0.25rem;
}
.info-label {
min-width: auto;
}
}
`]
})
export class MangaDetailComponent implements OnInit {
private readonly mangaService = inject(MangaService);
private readonly commentsService = inject(CommentsService);
readonly authService = inject(AuthService);
private readonly sanitizeService = inject(SanitizeService);
private readonly route = inject(ActivatedRoute);
private readonly router = inject(Router);
manga = signal<Manga | null>(null);
comments = signal<Comment[]>([]);
loading = signal(true);
commentsLoading = signal(false);
error = signal<string | null>(null);
newCommentContent = '';
showEditForm = signal(false);
ngOnInit() {
const mangaId = this.route.snapshot.paramMap.get('id');
if (!mangaId) {
this.error.set('No manga ID provided');
this.loading.set(false);
return;
}
this.loadManga(mangaId);
this.loadComments(mangaId);
}
private loadManga(mangaId: string) {
this.loading.set(true);
this.mangaService.getMangaById(mangaId).subscribe({
next: (manga) => {
if (!manga) {
this.error.set('Manga not found');
} else {
this.manga.set(manga);
}
this.loading.set(false);
},
error: () => {
this.error.set('Failed to load manga. It may not exist or there was an error.');
this.loading.set(false);
}
});
}
private loadComments(mangaId: string) {
this.commentsLoading.set(true);
this.commentsService.getCommentsForManga(mangaId).subscribe({
next: (comments) => {
this.comments.set(comments);
this.commentsLoading.set(false);
},
error: () => {
this.commentsLoading.set(false);
}
});
}
addComment() {
const manga = this.manga();
if (!manga || !this.newCommentContent.trim()) return;
this.commentsService.addCommentToManga(manga.id, { content: this.newCommentContent }).subscribe({
next: (comment) => {
this.comments.set([comment, ...this.comments()]);
this.newCommentContent = '';
}
});
}
handleCommentEdit(event: { commentId: string; content: string }) {
const manga = this.manga();
if (!manga) return;
this.commentsService.updateCommentOnManga(manga.id, event.commentId, event.content).subscribe({
next: (updatedComment) => {
this.comments.set(
this.comments().map(c => c.id === event.commentId ? updatedComment : c)
);
}
});
}
deleteComment(commentId: string) {
const manga = this.manga();
if (!manga || !confirm('Are you sure you want to delete this comment?')) return;
this.commentsService.deleteCommentFromManga(manga.id, commentId).subscribe({
next: () => {
this.comments.set(this.comments().filter(c => c.id !== commentId));
}
});
}
getStatusLabel(status: MangaStatus): string {
switch (status) {
case MangaStatus.reading: return 'Currently Reading';
case MangaStatus.completed: return 'Completed';
case MangaStatus.wantToRead: return 'Want to Read';
case MangaStatus.retired: return 'Retired';
}
}
formatDate(date: Date | string): string {
return new Date(date).toLocaleDateString('en-GB', {
year: 'numeric',
month: 'long',
day: 'numeric'
});
}
formatTimeSpent(minutes: number): string {
const hours = Math.floor(minutes / 60);
const mins = minutes % 60;
if (hours === 0) {
return `${mins} minutes`;
} else if (mins === 0) {
return `${hours} hour${hours === 1 ? '' : 's'}`;
} else {
return `${hours} hour${hours === 1 ? '' : 's'} ${mins} minute${mins === 1 ? '' : 's'}`;
}
}
toggleEditForm() {
this.showEditForm.update(value => !value);
}
saveEdit(data: UpdateMangaDto) {
const manga = this.manga();
if (!manga) return;
this.mangaService.updateManga(manga.id, data).subscribe({
next: (updatedManga) => {
this.manga.set(updatedManga);
this.showEditForm.set(false);
},
error: (err) => {
alert('Failed to update manga: ' + (err.error?.message || 'Unknown error'));
}
});
}
cancelEdit() {
this.showEditForm.set(false);
}
deleteManga() {
const manga = this.manga();
if (!manga) return;
if (!confirm(`Are you sure you want to delete "${manga.title}"? This action cannot be undone.`)) {
return;
}
this.mangaService.deleteManga(manga.id).subscribe({
next: () => {
this.router.navigate(['/manga']);
},
error: (err) => {
alert('Failed to delete manga: ' + (err.error?.message || 'Unknown error'));
}
});
}
}
@@ -7,7 +7,6 @@
import { Component, OnInit, inject, signal, computed } from '@angular/core';
import { CommonModule } from '@angular/common';
import { FormsModule } from '@angular/forms';
import { RouterLink } from '@angular/router';
import { MangaService } from '../../services/manga.service';
import { AuthService } from '../../services/auth.service';
import { CommentsService } from '../../services/comments.service';
@@ -20,13 +19,9 @@ import { Manga, MangaStatus, CreateMangaDto, UpdateMangaDto, Comment, Suggestion
@Component({
selector: 'app-manga-list',
standalone: true,
imports: [CommonModule, FormsModule, RouterLink, PaginationComponent, LikeButtonComponent],
imports: [CommonModule, FormsModule, PaginationComponent, LikeButtonComponent],
template: `
<div class="container">
<div class="page-hero">
<img src="/assets/avatars/manga-avatar.jpg" alt="Manga avatar" class="page-avatar" />
</div>
<div class="header-section">
<h2>My Manga Collection</h2>
@if (authService.isAdmin()) {
@@ -73,30 +68,9 @@ import { Manga, MangaStatus, CreateMangaDto, UpdateMangaDto, Comment, Suggestion
<option [value]="MangaStatus.reading">Currently Reading</option>
<option [value]="MangaStatus.completed">Completed</option>
<option [value]="MangaStatus.wantToRead">Want to Read</option>
<option [value]="MangaStatus.retired">Retired</option>
</select>
</div>
<div class="form-group">
<label for="dateStarted">Date Started</label>
<input
type="date"
id="dateStarted"
[(ngModel)]="newManga.dateStarted"
name="dateStarted"
>
</div>
<div class="form-group">
<label for="dateFinished">Date Finished</label>
<input
type="date"
id="dateFinished"
[(ngModel)]="newManga.dateFinished"
name="dateFinished"
>
</div>
<div class="form-group">
<label for="rating">Rating (1-10)</label>
<input
@@ -109,34 +83,6 @@ import { Manga, MangaStatus, CreateMangaDto, UpdateMangaDto, Comment, Suggestion
>
</div>
<div class="form-row">
<div class="form-group">
<label for="timeHours">Time Spent (Hours)</label>
<input
type="number"
id="timeHours"
[(ngModel)]="newMangaTimeHours"
name="timeHours"
min="0"
placeholder="0"
(ngModelChange)="updateNewMangaTimeSpent()"
>
</div>
<div class="form-group">
<label for="timeMinutes">Time Spent (Minutes)</label>
<input
type="number"
id="timeMinutes"
[(ngModel)]="newMangaTimeMinutes"
name="timeMinutes"
min="0"
max="59"
placeholder="0"
(ngModelChange)="updateNewMangaTimeSpent()"
>
</div>
</div>
<div class="form-group">
<label for="notes">Notes</label>
<textarea
@@ -169,7 +115,8 @@ import { Manga, MangaStatus, CreateMangaDto, UpdateMangaDto, Comment, Suggestion
</div>
<div class="form-group">
<div class="tags-input-container" aria-label="Tags">
<label>Tags</label>
<div class="tags-input-container">
@for (tag of newManga.tags; track tag; let i = $index) {
<span class="tag">
{{ tag }}
@@ -186,7 +133,8 @@ import { Manga, MangaStatus, CreateMangaDto, UpdateMangaDto, Comment, Suggestion
</div>
</div>
<div class="form-group" aria-label="External Links">
<div class="form-group">
<label>External Links</label>
<div class="links-list">
@for (link of newManga.links; track link.url; let i = $index) {
<div class="link-item">
@@ -252,30 +200,9 @@ import { Manga, MangaStatus, CreateMangaDto, UpdateMangaDto, Comment, Suggestion
<option [value]="MangaStatus.reading">Currently Reading</option>
<option [value]="MangaStatus.completed">Completed</option>
<option [value]="MangaStatus.wantToRead">Want to Read</option>
<option [value]="MangaStatus.retired">Retired</option>
</select>
</div>
<div class="form-group">
<label for="edit-dateStarted">Date Started</label>
<input
type="date"
id="edit-dateStarted"
[(ngModel)]="editManga.dateStarted"
name="dateStarted"
>
</div>
<div class="form-group">
<label for="edit-dateFinished">Date Finished</label>
<input
type="date"
id="edit-dateFinished"
[(ngModel)]="editManga.dateFinished"
name="dateFinished"
>
</div>
<div class="form-group">
<label for="edit-rating">Rating (1-10)</label>
<input
@@ -288,34 +215,6 @@ import { Manga, MangaStatus, CreateMangaDto, UpdateMangaDto, Comment, Suggestion
>
</div>
<div class="form-row">
<div class="form-group">
<label for="edit-timeHours">Time Spent (Hours)</label>
<input
type="number"
id="edit-timeHours"
[(ngModel)]="editMangaTimeHours"
name="timeHours"
min="0"
placeholder="0"
(ngModelChange)="updateEditMangaTimeSpent()"
>
</div>
<div class="form-group">
<label for="edit-timeMinutes">Time Spent (Minutes)</label>
<input
type="number"
id="edit-timeMinutes"
[(ngModel)]="editMangaTimeMinutes"
name="timeMinutes"
min="0"
max="59"
placeholder="0"
(ngModelChange)="updateEditMangaTimeSpent()"
>
</div>
</div>
<div class="form-group">
<label for="edit-notes">Notes</label>
<textarea
@@ -348,7 +247,8 @@ import { Manga, MangaStatus, CreateMangaDto, UpdateMangaDto, Comment, Suggestion
</div>
<div class="form-group">
<div class="tags-input-container" aria-label="Tags">
<label>Tags</label>
<div class="tags-input-container">
@for (tag of editManga.tags; track tag; let i = $index) {
<span class="tag">
{{ tag }}
@@ -365,7 +265,8 @@ import { Manga, MangaStatus, CreateMangaDto, UpdateMangaDto, Comment, Suggestion
</div>
</div>
<div class="form-group" aria-label="External Links">
<div class="form-group">
<label>External Links</label>
<div class="links-list">
@for (link of editManga.links; track link.url; let i = $index) {
<div class="link-item">
@@ -538,13 +439,6 @@ import { Manga, MangaStatus, CreateMangaDto, UpdateMangaDto, Comment, Suggestion
>
Want to Read ({{ wantToReadCount() }})
</button>
<button
(click)="setFilter(MangaStatus.retired)"
[class.active]="statusFilter() === MangaStatus.retired"
class="filter-btn"
>
Retired ({{ retiredCount() }})
</button>
</div>
@if (loading()) {
@@ -565,34 +459,54 @@ import { Manga, MangaStatus, CreateMangaDto, UpdateMangaDto, Comment, Suggestion
<div class="manga-grid">
@for (manga of paginatedManga(); track manga.id) {
<div class="manga-card" [class.completed]="manga.status === MangaStatus.completed">
<a [routerLink]="['/manga', manga.id]" class="card-link">
<img [src]="manga.coverImage || '/assets/default-cover.jpg'" [alt]="manga.title" class="manga-cover">
@if (manga.coverImage) {
<img [src]="manga.coverImage" [alt]="manga.title" class="manga-cover">
}
<div class="manga-info">
<h3>{{ manga.title }}</h3>
<p class="author">by {{ manga.author }}</p>
<span class="status status-{{ manga.status }}">
{{ getStatusLabel(manga.status) }}
</span>
<div class="manga-info">
<h3>{{ manga.title }}</h3>
<p class="author">by {{ manga.author }}</p>
<span class="status status-{{ manga.status }}">
{{ getStatusLabel(manga.status) }}
</span>
@if (manga.rating) {
<div class="rating">
@for (star of [1,2,3,4,5,6,7,8,9,10]; track star) {
<span [class.filled]="star <= manga.rating!">★</span>
}
</div>
}
</div>
</a>
@if (manga.rating) {
<div class="rating">
@for (star of [1,2,3,4,5,6,7,8,9,10]; track star) {
<span [class.filled]="star <= manga.rating!">★</span>
}
</div>
}
<div class="card-actions">
<app-like-button
entityType="manga"
[entityId]="manga.id"
></app-like-button>
@if (manga.notes) {
<p class="notes">{{ manga.notes }}</p>
}
@if (manga.tags && manga.tags.length > 0) {
<div class="tags-display">
@for (tag of manga.tags; track tag) {
<span class="tag-chip">{{ tag }}</span>
}
</div>
}
@if (manga.links && manga.links.length > 0) {
<div class="links-display">
@for (link of manga.links; track link.url) {
<a [href]="link.url" target="_blank" rel="noopener noreferrer" class="external-link">
{{ link.title }} ↗
</a>
}
</div>
}
@if (authService.isAdmin()) {
<div class="admin-actions">
<div class="actions">
<button (click)="startEdit(manga)" class="btn btn-secondary btn-sm">
Edit
</button>
@@ -601,6 +515,85 @@ import { Manga, MangaStatus, CreateMangaDto, UpdateMangaDto, Comment, Suggestion
</button>
</div>
}
<div class="comments-section">
<button (click)="toggleComments(manga.id)" class="btn btn-secondary btn-sm comments-toggle">
{{ expandedComments()[manga.id] ? 'Hide' : 'Show' }} Comments{{ comments()[manga.id] ? ' (' + getCommentCount(manga.id) + ')' : '' }}
</button>
@if (expandedComments()[manga.id]) {
<div class="comments-container">
@if (authService.isAuthenticated()) {
@if (authService.user()?.isBanned) {
<div class="banned-notice">
You have been banned from commenting.
</div>
} @else {
<form (ngSubmit)="addComment(manga.id)" class="comment-form">
<textarea
[(ngModel)]="newCommentContent[manga.id]"
name="comment"
placeholder="Add a comment (Markdown supported)..."
rows="2"
></textarea>
<button type="submit" class="btn btn-primary btn-sm">Post Comment</button>
</form>
}
}
@if (commentsLoading()[manga.id]) {
<div class="comments-loading">Loading comments...</div>
} @else {
@for (comment of comments()[manga.id] || []; track comment.id) {
<div class="comment">
<div class="comment-header">
@if (comment.user.avatar) {
<img [src]="comment.user.avatar" [alt]="comment.user.username" class="comment-avatar">
}
<span class="comment-author">{{ comment.user.username }}</span>
@if (comment.user.inDiscord) {
<span class="discord-badge">Discord</span>
}
@if (comment.user.isVip) {
<span class="vip-badge">VIP</span>
}
@if (comment.user.isMod) {
<span class="mod-badge">Mod</span>
}
@if (comment.user.isStaff) {
<span class="staff-badge">Staff</span>
}
<span class="comment-date">{{ formatDate(comment.createdAt) }}</span>
@if (canEditComment(comment)) {
<button (click)="startEditComment(manga.id, comment)" class="btn btn-secondary btn-xs">Edit</button>
}
@if (canDeleteComment(comment)) {
<button (click)="deleteComment(manga.id, comment.id)" class="btn btn-danger btn-xs">Delete</button>
}
</div>
@if (editingCommentId() === comment.id) {
<div class="comment-edit-form">
<textarea
[(ngModel)]="editCommentContent"
name="editComment"
rows="3"
></textarea>
<div class="comment-edit-actions">
<button (click)="saveCommentEdit(manga.id, comment.id)" class="btn btn-primary btn-xs">Save</button>
<button (click)="cancelCommentEdit()" class="btn btn-secondary btn-xs">Cancel</button>
</div>
</div>
} @else {
<div class="comment-content" [innerHTML]="sanitizeService.sanitizeHtml(comment.content)"></div>
}
</div>
} @empty {
<div class="no-comments">No comments yet. Be the first to comment!</div>
}
}
</div>
}
</div>
</div>
</div>
}
@@ -623,26 +616,6 @@ import { Manga, MangaStatus, CreateMangaDto, UpdateMangaDto, Comment, Suggestion
padding: 2rem;
}
.page-hero {
text-align: center;
margin-bottom: 2rem;
}
.page-avatar {
width: 120px;
height: 120px;
border-radius: 50%;
object-fit: cover;
border: 3px solid #00b894;
box-shadow: 0 4px 12px rgba(0, 184, 148, 0.3);
transition: all 0.3s;
}
.page-avatar:hover {
transform: scale(1.05);
box-shadow: 0 8px 16px rgba(0, 184, 148, 0.5);
}
.header-section {
display: flex;
justify-content: space-between;
@@ -693,13 +666,6 @@ import { Manga, MangaStatus, CreateMangaDto, UpdateMangaDto, Comment, Suggestion
font-size: 1rem;
}
.form-row {
display: grid;
grid-template-columns: 1fr 1fr;
gap: 1rem;
margin-bottom: 1rem;
}
.form-actions {
display: flex;
gap: 1rem;
@@ -831,8 +797,6 @@ import { Manga, MangaStatus, CreateMangaDto, UpdateMangaDto, Comment, Suggestion
border-radius: 8px;
overflow: hidden;
transition: transform 0.3s, box-shadow 0.3s;
display: flex;
flex-direction: column;
}
.manga-card:hover {
@@ -844,13 +808,6 @@ import { Manga, MangaStatus, CreateMangaDto, UpdateMangaDto, Comment, Suggestion
opacity: 0.8;
}
.card-link {
text-decoration: none;
color: inherit;
display: block;
flex: 1;
}
.manga-cover {
width: 100%;
height: 200px;
@@ -864,7 +821,6 @@ import { Manga, MangaStatus, CreateMangaDto, UpdateMangaDto, Comment, Suggestion
.manga-info h3 {
margin: 0 0 0.5rem 0;
font-size: 1.1rem;
color: #1f2937;
}
.author {
@@ -880,7 +836,6 @@ import { Manga, MangaStatus, CreateMangaDto, UpdateMangaDto, Comment, Suggestion
border-radius: 4px;
font-size: 0.8rem;
font-weight: 500;
margin: 0.5rem 0;
}
.status-READING { background: #fef3c7; color: #92400e; }
@@ -900,18 +855,14 @@ import { Manga, MangaStatus, CreateMangaDto, UpdateMangaDto, Comment, Suggestion
color: #ec4899;
}
.card-actions {
padding: 0.75rem 1rem;
border-top: 1px solid #e5e7eb;
display: flex;
align-items: center;
justify-content: space-between;
gap: 0.5rem;
.notes {
font-size: 0.9rem;
color: #4b5563;
margin: 0.5rem 0;
}
.admin-actions {
display: flex;
gap: 0.5rem;
.actions {
margin-top: 1rem;
}
.btn {
@@ -933,6 +884,145 @@ import { Manga, MangaStatus, CreateMangaDto, UpdateMangaDto, Comment, Suggestion
.btn-sm { padding: 0.25rem 0.75rem; font-size: 0.85rem; }
.btn-xs { padding: 0.15rem 0.5rem; font-size: 0.75rem; }
.comments-section {
margin-top: 1rem;
border-top: 1px solid #e5e7eb;
padding-top: 1rem;
}
.comments-toggle {
width: 100%;
}
.comments-container {
margin-top: 1rem;
}
.comment-form {
margin-bottom: 1rem;
}
.comment-form textarea {
width: 100%;
padding: 0.5rem;
border: 1px solid #ddd;
border-radius: 4px;
font-size: 0.9rem;
resize: vertical;
margin-bottom: 0.5rem;
}
.comment {
background: #f8f9fa;
border: 1px solid #e5e7eb;
border-radius: 4px;
padding: 0.75rem;
margin-bottom: 0.5rem;
}
.comment-header {
display: flex;
align-items: center;
gap: 0.5rem;
margin-bottom: 0.5rem;
flex-wrap: wrap;
}
.comment-avatar {
width: 24px;
height: 24px;
border-radius: 50%;
}
.comment-author {
font-weight: 500;
color: #374151;
}
.discord-badge {
background: #5865f2;
color: white;
padding: 0.1rem 0.4rem;
border-radius: 3px;
font-size: 0.7rem;
font-weight: 500;
}
.vip-badge {
background: linear-gradient(135deg, #ffd700, #ffaa00);
color: #1a1a1a;
padding: 0.1rem 0.4rem;
border-radius: 3px;
font-size: 0.7rem;
font-weight: 600;
}
.mod-badge {
background: linear-gradient(135deg, #00b894, #00cec9);
color: white;
padding: 0.1rem 0.4rem;
border-radius: 3px;
font-size: 0.7rem;
font-weight: 600;
}
.staff-badge {
background: linear-gradient(135deg, #e84393, #fd79a8);
color: white;
padding: 0.1rem 0.4rem;
border-radius: 3px;
font-size: 0.7rem;
font-weight: 600;
}
.comment-date {
font-size: 0.75rem;
color: #6b7280;
}
.comment-content {
font-size: 0.9rem;
color: #4b5563;
}
.comments-loading,
.no-comments {
text-align: center;
padding: 1rem;
color: #6b7280;
font-size: 0.9rem;
}
.banned-notice {
background: #fef2f2;
border: 1px solid #fecaca;
color: #991b1b;
padding: 0.75rem 1rem;
border-radius: 4px;
text-align: center;
margin-bottom: 1rem;
font-size: 0.9rem;
}
.comment-edit-form {
margin-top: 0.5rem;
}
.comment-edit-form textarea {
width: 100%;
padding: 0.5rem;
border: 1px solid #ddd;
border-radius: 4px;
font-size: 0.9rem;
resize: vertical;
}
.comment-edit-actions {
display: flex;
gap: 0.5rem;
margin-top: 0.5rem;
}
.image-preview {
margin-top: 0.5rem;
display: flex;
@@ -1014,6 +1104,25 @@ import { Manga, MangaStatus, CreateMangaDto, UpdateMangaDto, Comment, Suggestion
opacity: 0.8;
}
.tags-display {
display: flex;
flex-wrap: wrap;
gap: 0.5rem;
margin: 0.5rem 0;
}
.tag-chip {
background: rgba(0, 184, 148, 0.2);
color: #00b894;
padding: 0.2rem 0.5rem;
border-radius: 4px;
font-size: 0.8rem;
}
.links-list {
margin-bottom: 0.5rem;
}
.link-item {
display: flex;
justify-content: space-between;
@@ -1034,6 +1143,28 @@ import { Manga, MangaStatus, CreateMangaDto, UpdateMangaDto, Comment, Suggestion
flex: 1;
min-width: 120px;
}
.links-display {
display: flex;
flex-wrap: wrap;
gap: 0.5rem;
margin: 0.5rem 0;
}
.external-link {
color: #00b894;
text-decoration: none;
font-size: 0.85rem;
padding: 0.2rem 0.5rem;
background: rgba(0, 184, 148, 0.1);
border-radius: 4px;
transition: background 0.2s;
}
.external-link:hover {
background: rgba(0, 184, 148, 0.2);
text-decoration: underline;
}
`]
})
export class MangaListComponent implements OnInit {
@@ -1085,7 +1216,6 @@ export class MangaListComponent implements OnInit {
readingCount = computed(() => this.mangaList().filter(m => m.status === MangaStatus.reading).length);
completedCount = computed(() => this.mangaList().filter(m => m.status === MangaStatus.completed).length);
wantToReadCount = computed(() => this.mangaList().filter(m => m.status === MangaStatus.wantToRead).length);
retiredCount = computed(() => this.mangaList().filter(m => m.status === MangaStatus.retired).length);
allTags = computed(() => {
const tagsSet = new Set<string>();
@@ -1134,12 +1264,10 @@ export class MangaListComponent implements OnInit {
totalFilteredManga = computed(() => this.filteredManga().length);
newManga: Partial<CreateMangaDto> & { dateStarted?: Date; dateFinished?: Date } = {
newManga: Partial<CreateMangaDto> = {
title: '',
author: '',
status: MangaStatus.wantToRead,
dateStarted: undefined,
dateFinished: undefined,
rating: undefined,
notes: '',
tags: [],
@@ -1148,12 +1276,6 @@ export class MangaListComponent implements OnInit {
editManga: Partial<UpdateMangaDto> = {};
// Time tracking state
newMangaTimeHours = 0;
newMangaTimeMinutes = 0;
editMangaTimeHours = 0;
editMangaTimeMinutes = 0;
// Tags and links input state
newTagInput = '';
editTagInput = '';
@@ -1222,7 +1344,6 @@ export class MangaListComponent implements OnInit {
case MangaStatus.reading: return 'Currently Reading';
case MangaStatus.completed: return 'Completed';
case MangaStatus.wantToRead: return 'Want to Read';
case MangaStatus.retired: return 'Retired';
}
}
@@ -1238,16 +1359,12 @@ export class MangaListComponent implements OnInit {
title: '',
author: '',
status: MangaStatus.wantToRead,
dateStarted: undefined,
dateFinished: undefined,
rating: undefined,
notes: '',
coverImage: undefined,
tags: [],
links: []
};
this.newMangaTimeHours = 0;
this.newMangaTimeMinutes = 0;
this.newMangaImagePreview.set(null);
this.imageError.set(null);
this.newTagInput = '';
@@ -1255,16 +1372,6 @@ export class MangaListComponent implements OnInit {
this.newLinkUrl = '';
}
updateNewMangaTimeSpent() {
const totalMinutes = (this.newMangaTimeHours * 60) + this.newMangaTimeMinutes;
this.newManga.timeSpent = totalMinutes > 0 ? totalMinutes : undefined;
}
updateEditMangaTimeSpent() {
const totalMinutes = (this.editMangaTimeHours * 60) + this.editMangaTimeMinutes;
this.editManga.timeSpent = totalMinutes > 0 ? totalMinutes : undefined;
}
addTag(target: 'new' | 'edit') {
const input = target === 'new' ? this.newTagInput.trim() : this.editTagInput.trim();
if (!input) return;
@@ -1317,8 +1424,6 @@ export class MangaListComponent implements OnInit {
title: this.newManga.title,
author: this.newManga.author,
status: this.newManga.status,
dateStarted: this.newManga.dateStarted ? new Date(this.newManga.dateStarted) : undefined,
dateFinished: this.newManga.dateFinished ? new Date(this.newManga.dateFinished) : undefined,
rating: this.newManga.rating,
notes: this.newManga.notes,
coverImage: this.newManga.coverImage,
@@ -1346,32 +1451,18 @@ export class MangaListComponent implements OnInit {
title: manga.title,
author: manga.author,
status: manga.status,
dateStarted: manga.dateStarted,
dateFinished: manga.dateFinished,
rating: manga.rating,
notes: manga.notes,
coverImage: manga.coverImage,
tags: [...(manga.tags || [])],
links: [...(manga.links || [])],
timeSpent: manga.timeSpent
links: [...(manga.links || [])]
};
// Populate time fields from existing timeSpent
if (manga.timeSpent) {
this.editMangaTimeHours = Math.floor(manga.timeSpent / 60);
this.editMangaTimeMinutes = manga.timeSpent % 60;
} else {
this.editMangaTimeHours = 0;
this.editMangaTimeMinutes = 0;
}
this.editMangaImagePreview.set(manga.coverImage || null);
this.showAddForm.set(false);
this.imageError.set(null);
this.editTagInput = '';
this.editLinkTitle = '';
this.editLinkUrl = '';
// Scroll to top so the form is visible
window.scrollTo({ top: 0, behavior: 'smooth' });
}
cancelEdit() {
@@ -1388,13 +1479,7 @@ export class MangaListComponent implements OnInit {
const manga = this.editingManga();
if (!manga || !this.editManga.title || !this.editManga.author || !this.editManga.status) return;
const updateData = {
...this.editManga,
dateStarted: this.editManga.dateStarted ? new Date(this.editManga.dateStarted) : undefined,
dateFinished: this.editManga.dateFinished ? new Date(this.editManga.dateFinished) : undefined,
};
this.mangaService.updateManga(manga.id, updateData).subscribe(() => {
this.mangaService.updateManga(manga.id, this.editManga).subscribe(() => {
this.loadManga();
this.cancelEdit();
});
@@ -1455,19 +1540,6 @@ export class MangaListComponent implements OnInit {
return new Date(date).toLocaleDateString();
}
formatTimeSpent(minutes: number): string {
const hours = Math.floor(minutes / 60);
const mins = minutes % 60;
if (hours === 0) {
return `${mins}m`;
} else if (mins === 0) {
return `${hours}h`;
} else {
return `${hours}h ${mins}m`;
}
}
toggleComments(mangaId: string) {
const expanded = this.expandedComments();
const isCurrentlyExpanded = expanded[mangaId];
@@ -1602,7 +1674,7 @@ export class MangaListComponent implements OnInit {
try {
await this.suggestionService.createSuggestion({
entityType: SuggestionEntity.manga,
entityType: SuggestionEntity.MANGA,
title: this.suggestedManga.title,
author: this.suggestedManga.author,
notes: this.suggestedManga.notes,
@@ -1614,21 +1686,4 @@ export class MangaListComponent implements OnInit {
alert('Failed to submit suggestion. Please try again.');
}
}
handleCommentEdit(mangaId: string, event: { commentId: string; content: string }) {
this.commentsService.updateCommentOnManga(mangaId, event.commentId, event.content).subscribe({
next: (updatedComment) => {
this.comments.set({
...this.comments(),
[mangaId]: (this.comments()[mangaId] || []).map(c =>
c.id === event.commentId ? updatedComment : c
)
});
}
});
}
getCommentsSignal(mangaId: string) {
return signal(this.comments()[mangaId] || []);
}
}
@@ -1,781 +0,0 @@
/**
* @copyright 2026 NHCarrigan
* @license Naomi's Public License
* @author Naomi Carrigan
*/
import { Component, OnInit, inject, signal } from '@angular/core';
import { CommonModule } from '@angular/common';
import { RouterLink, ActivatedRoute, Router } from '@angular/router';
import { FormsModule } from '@angular/forms';
import { MusicService } from '../../services/music.service';
import { CommentsService } from '../../services/comments.service';
import { AuthService } from '../../services/auth.service';
import { SanitizeService } from '../../services/sanitize.service';
import { CommentDisplayComponent } from '../comment-display/comment-display.component';
import { LikeButtonComponent } from '../shared/like-button.component';
import { MusicFormComponent } from '../shared/music-form.component';
import { Music, Comment, MusicStatus, MusicType, UpdateMusicDto } from '@library/shared-types';
@Component({
selector: 'app-music-detail',
standalone: true,
imports: [CommonModule, RouterLink, FormsModule, CommentDisplayComponent, LikeButtonComponent, MusicFormComponent],
template: `
<div class="container">
<div class="breadcrumb">
<a routerLink="/music" class="breadcrumb-link">← Back to Music</a>
</div>
@if (showEditForm() && authService.user()?.isAdmin && music()) {
<app-music-form
mode="edit"
[music]="music()!"
(formSubmit)="saveEdit($event)"
(formCancel)="cancelEdit()"
></app-music-form>
}
@if (loading()) {
<div class="loading">Loading music details...</div>
} @else if (error()) {
<div class="error-state">
<h2>Music Not Found</h2>
<p>{{ error() }}</p>
<a routerLink="/music" class="btn btn-primary">Return to Music</a>
</div>
} @else if (music()) {
<div class="music-detail-card">
<div class="music-cover-section">
<img [src]="music()!.coverArt || '/assets/default-cover.jpg'" [alt]="music()!.title" class="music-cover-large">
</div>
<div class="music-content">
<div class="music-header">
<h1>{{ music()!.title }}</h1>
<span class="type-badge type-{{ music()!.type }}">
{{ getTypeLabel(music()!.type) }}
</span>
<span class="status status-{{ music()!.status }}">
{{ getStatusLabel(music()!.status) }}
</span>
</div>
<p class="artist">by {{ music()!.artist }}</p>
@if (authService.user()?.isAdmin) {
<div class="admin-actions">
<button (click)="editMusic()" class="btn btn-edit">✏️ Edit</button>
<button (click)="deleteMusic()" class="btn btn-delete">🗑️ Delete</button>
</div>
}
@if (music()!.rating) {
<div class="info-row">
<span class="info-label">Rating:</span>
<div class="rating">
@for (star of [1,2,3,4,5,6,7,8,9,10]; track star) {
<span [class.filled]="star <= music()!.rating!">★</span>
}
<span class="rating-text">({{ music()!.rating }}/10)</span>
</div>
</div>
}
@if (music()!.timeSpent) {
<div class="info-row">
<span class="info-label">Listening Time:</span>
<span class="info-value time-spent">{{ formatTimeSpent(music()!.timeSpent!) }}</span>
</div>
}
@if (music()!.dateStarted) {
<div class="info-row">
<span class="info-label">Started:</span>
<span class="info-value">{{ formatDate(music()!.dateStarted!) }}</span>
</div>
}
@if (music()!.dateFinished) {
<div class="info-row">
<span class="info-label">Finished:</span>
<span class="info-value">{{ formatDate(music()!.dateFinished!) }}</span>
</div>
}
<div class="info-row">
<span class="info-label">Added:</span>
<span class="info-value">{{ formatDate(music()!.createdAt) }}</span>
</div>
<div class="info-row">
<span class="info-label">Last Updated:</span>
<span class="info-value">{{ formatDate(music()!.updatedAt) }}</span>
</div>
<div class="like-section">
<app-like-button
entityType="music"
[entityId]="music()!.id"
></app-like-button>
</div>
@if (music()!.notes) {
<div class="notes-section">
<h3>Notes</h3>
<p class="notes">{{ music()!.notes }}</p>
</div>
}
@if (music()!.tags && music()!.tags.length > 0) {
<div class="tags-section">
<h3>Tags</h3>
<div class="tags-display">
@for (tag of music()!.tags; track tag) {
<span class="tag-chip">{{ tag }}</span>
}
</div>
</div>
}
@if (music()!.links && music()!.links.length > 0) {
<div class="links-section">
<h3>External Links</h3>
<div class="links-display">
@for (link of music()!.links; track link.url) {
<a [href]="link.url" target="_blank" rel="noopener noreferrer" class="external-link">
{{ link.title }} ↗
</a>
}
</div>
</div>
}
<div class="comments-section">
<h3>Comments</h3>
@if (authService.isAuthenticated()) {
@if (authService.user()?.isBanned) {
<div class="banned-notice">
You have been banned from commenting.
</div>
} @else {
<form (ngSubmit)="addComment()" class="comment-form">
<textarea
[(ngModel)]="newCommentContent"
name="comment"
placeholder="Add a comment (Markdown supported)..."
rows="3"
></textarea>
<button type="submit" class="btn btn-primary">Post Comment</button>
</form>
}
} @else {
<div class="auth-prompt">
<p>Please sign in to comment.</p>
</div>
}
@if (commentsLoading()) {
<div class="comments-loading">Loading comments...</div>
} @else {
<app-comment-display
[comments]="comments"
(edit)="handleCommentEdit($event)"
(delete)="deleteComment($event)"
/>
}
</div>
</div>
</div>
}
</div>
`,
styles: [`
.container {
max-width: 900px;
margin: 0 auto;
padding: 2rem;
}
.breadcrumb {
margin-bottom: 1.5rem;
}
.breadcrumb-link {
color: #74b9ff;
text-decoration: none;
font-weight: 500;
transition: opacity 0.3s;
}
.breadcrumb-link:hover {
opacity: 0.8;
}
.loading {
text-align: center;
padding: 3rem;
color: #666;
font-size: 1.1rem;
}
.error-state {
text-align: center;
padding: 3rem;
background: #fef2f2;
border: 1px solid #fecaca;
border-radius: 8px;
}
.error-state h2 {
color: #991b1b;
margin-bottom: 1rem;
}
.error-state p {
color: #dc2626;
margin-bottom: 1.5rem;
}
.music-detail-card {
background: white;
border: 1px solid #e5e7eb;
border-radius: 8px;
overflow: hidden;
box-shadow: 0 1px 3px rgba(0,0,0,0.1);
}
.music-cover-section {
width: 100%;
background: #f3f4f6;
display: flex;
justify-content: center;
align-items: center;
padding: 2rem;
}
.music-cover-large {
max-width: 100%;
max-height: 400px;
object-fit: contain;
border-radius: 4px;
}
.music-content {
padding: 2rem;
}
.music-header {
display: flex;
align-items: center;
gap: 1rem;
margin-bottom: 1rem;
flex-wrap: wrap;
}
.music-header h1 {
margin: 0;
font-size: 2rem;
color: #1f2937;
}
.artist {
color: #6b7280;
font-weight: 500;
font-size: 1.1rem;
margin: 0 0 1rem 0;
}
.type-badge {
display: inline-block;
padding: 0.5rem 1rem;
border-radius: 4px;
font-size: 0.9rem;
font-weight: 500;
}
.type-album {
background: #a78bfa;
color: white;
}
.type-single {
background: #fb923c;
color: white;
}
.type-ep {
background: #60a5fa;
color: white;
}
.status {
display: inline-block;
padding: 0.5rem 1rem;
border-radius: 4px;
font-size: 0.9rem;
font-weight: 500;
}
.status-listening {
background: #fef3c7;
color: #92400e;
}
.status-completed {
background: #d1fae5;
color: #065f46;
}
.status-wantToListen {
background: #e0e7ff;
color: #3730a3;
}
.status-retired {
background: #f3f4f6;
color: #4b5563;
}
.admin-actions {
display: flex;
gap: 0.75rem;
margin-bottom: 1.5rem;
padding: 1rem;
background: #fef3c7;
border-radius: 4px;
border: 1px solid #fbbf24;
}
.btn-edit {
background: #3b82f6;
color: white;
}
.btn-edit:hover {
background: #2563eb;
}
.btn-delete {
background: #ef4444;
color: white;
}
.btn-delete:hover {
background: #dc2626;
}
.info-row {
display: flex;
align-items: center;
gap: 1rem;
margin-bottom: 0.75rem;
padding: 0.5rem 0;
border-bottom: 1px solid #f3f4f6;
}
.info-label {
font-weight: 600;
color: #4b5563;
min-width: 120px;
}
.info-value {
color: #1f2937;
}
.time-spent {
color: #8b5cf6;
font-weight: 600;
}
.rating {
display: flex;
align-items: center;
gap: 0.25rem;
}
.rating span {
color: #e5e7eb;
font-size: 1.2rem;
}
.rating span.filled {
color: #f59e0b;
}
.rating-text {
margin-left: 0.5rem;
font-size: 1rem;
color: #6b7280;
font-weight: 500;
}
.like-section {
margin: 1.5rem 0;
}
.notes-section,
.tags-section,
.links-section {
margin-top: 2rem;
}
.notes-section h3,
.tags-section h3,
.links-section h3,
.comments-section h3 {
font-size: 1.25rem;
color: #1f2937;
margin-bottom: 1rem;
}
.notes {
font-size: 1rem;
color: #4b5563;
line-height: 1.6;
white-space: pre-wrap;
}
.tags-display {
display: flex;
flex-wrap: wrap;
gap: 0.5rem;
}
.tag-chip {
background: #74b9ff;
color: white;
padding: 0.375rem 0.875rem;
border-radius: 12px;
font-size: 0.875rem;
font-weight: 500;
}
.links-display {
display: flex;
flex-wrap: wrap;
gap: 0.75rem;
}
.external-link {
color: #74b9ff;
text-decoration: none;
font-size: 0.95rem;
padding: 0.5rem 1rem;
border: 2px solid #74b9ff;
border-radius: 4px;
transition: all 0.2s;
font-weight: 500;
}
.external-link:hover {
background: #74b9ff;
color: white;
}
.comments-section {
margin-top: 2rem;
padding-top: 2rem;
border-top: 2px solid #e5e7eb;
}
.comment-form {
margin-bottom: 1.5rem;
}
.comment-form textarea {
width: 100%;
padding: 0.75rem;
border: 1px solid #d1d5db;
border-radius: 4px;
font-size: 1rem;
resize: vertical;
margin-bottom: 0.75rem;
font-family: inherit;
}
.comment-form textarea:focus {
outline: none;
border-color: #74b9ff;
}
.banned-notice {
background: #fef2f2;
border: 1px solid #fecaca;
color: #991b1b;
padding: 1rem;
border-radius: 4px;
text-align: center;
margin-bottom: 1.5rem;
font-size: 0.95rem;
}
.auth-prompt {
background: #f3f4f6;
padding: 1rem;
border-radius: 4px;
text-align: center;
margin-bottom: 1.5rem;
}
.auth-prompt p {
margin: 0;
color: #4b5563;
}
.comments-loading {
text-align: center;
padding: 1.5rem;
color: #6b7280;
font-size: 0.95rem;
}
.btn {
padding: 0.625rem 1.25rem;
border: none;
border-radius: 4px;
cursor: pointer;
font-size: 1rem;
transition: all 0.3s;
font-weight: 500;
}
.btn:hover {
opacity: 0.9;
}
.btn-primary {
background: #74b9ff;
color: white;
}
@media (max-width: 640px) {
.container {
padding: 1rem;
}
.music-header {
flex-direction: column;
align-items: flex-start;
}
.music-header h1 {
font-size: 1.5rem;
}
.admin-actions {
display: flex;
gap: 0.75rem;
margin-bottom: 1.5rem;
padding: 1rem;
background: #fef3c7;
border-radius: 4px;
border: 1px solid #fbbf24;
}
.btn-edit {
background: #3b82f6;
color: white;
}
.btn-edit:hover {
background: #2563eb;
}
.btn-delete {
background: #ef4444;
color: white;
}
.btn-delete:hover {
background: #dc2626;
}
.info-row {
flex-direction: column;
align-items: flex-start;
gap: 0.25rem;
}
.info-label {
min-width: auto;
}
}
`]
})
export class MusicDetailComponent implements OnInit {
private readonly musicService = inject(MusicService);
private readonly commentsService = inject(CommentsService);
readonly authService = inject(AuthService);
private readonly sanitizeService = inject(SanitizeService);
private readonly route = inject(ActivatedRoute);
private readonly router = inject(Router);
music = signal<Music | null>(null);
comments = signal<Comment[]>([]);
loading = signal(true);
commentsLoading = signal(false);
error = signal<string | null>(null);
newCommentContent = '';
showEditForm = signal(false);
ngOnInit() {
const musicId = this.route.snapshot.paramMap.get('id');
if (!musicId) {
this.error.set('No music ID provided');
this.loading.set(false);
return;
}
this.loadMusic(musicId);
this.loadComments(musicId);
}
private loadMusic(musicId: string) {
this.loading.set(true);
this.musicService.getMusicById(musicId).subscribe({
next: (music) => {
if (!music) {
this.error.set('Music not found');
} else {
this.music.set(music);
}
this.loading.set(false);
},
error: () => {
this.error.set('Failed to load music. It may not exist or there was an error.');
this.loading.set(false);
}
});
}
private loadComments(musicId: string) {
this.commentsLoading.set(true);
this.commentsService.getCommentsForMusic(musicId).subscribe({
next: (comments) => {
this.comments.set(comments);
this.commentsLoading.set(false);
},
error: () => {
this.commentsLoading.set(false);
}
});
}
addComment() {
const music = this.music();
if (!music || !this.newCommentContent.trim()) return;
this.commentsService.addCommentToMusic(music.id, { content: this.newCommentContent }).subscribe({
next: (comment) => {
this.comments.set([comment, ...this.comments()]);
this.newCommentContent = '';
}
});
}
handleCommentEdit(event: { commentId: string; content: string }) {
const music = this.music();
if (!music) return;
this.commentsService.updateCommentOnMusic(music.id, event.commentId, event.content).subscribe({
next: (updatedComment) => {
this.comments.set(
this.comments().map(c => c.id === event.commentId ? updatedComment : c)
);
}
});
}
deleteComment(commentId: string) {
const music = this.music();
if (!music || !confirm('Are you sure you want to delete this comment?')) return;
this.commentsService.deleteCommentFromMusic(music.id, commentId).subscribe({
next: () => {
this.comments.set(this.comments().filter(c => c.id !== commentId));
}
});
}
getTypeLabel(type: MusicType): string {
switch (type) {
case MusicType.album: return 'Album';
case MusicType.single: return 'Single';
case MusicType.ep: return 'EP';
}
}
getStatusLabel(status: MusicStatus): string {
switch (status) {
case MusicStatus.listening: return 'Currently Listening';
case MusicStatus.completed: return 'Completed';
case MusicStatus.wantToListen: return 'Want to Listen';
case MusicStatus.retired: return 'Retired';
}
}
formatDate(date: Date | string): string {
return new Date(date).toLocaleDateString('en-GB', {
year: 'numeric',
month: 'long',
day: 'numeric'
});
}
formatTimeSpent(minutes: number): string {
const hours = Math.floor(minutes / 60);
const mins = minutes % 60;
if (hours === 0) {
return `${mins} minutes`;
} else if (mins === 0) {
return `${hours} hour${hours === 1 ? '' : 's'}`;
} else {
return `${hours} hour${hours === 1 ? '' : 's'} ${mins} minute${mins === 1 ? '' : 's'}`;
}
}
editMusic() {
this.showEditForm.set(true);
}
cancelEdit() {
this.showEditForm.set(false);
}
saveEdit(data: UpdateMusicDto) {
const music = this.music();
if (!music) return;
this.musicService.updateMusic(music.id, data).subscribe({
next: (updatedMusic) => {
this.music.set(updatedMusic);
this.showEditForm.set(false);
},
error: (err) => {
alert('Failed to update music: ' + (err.error?.message || 'Unknown error'));
}
});
}
deleteMusic() {
const music = this.music();
if (!music) return;
if (!confirm(`Are you sure you want to delete "${music.title}"? This action cannot be undone.`)) {
return;
}
this.musicService.deleteMusic(music.id).subscribe({
next: () => {
this.router.navigate(['/music']);
},
error: (err) => {
alert('Failed to delete music: ' + (err.error?.message || 'Unknown error'));
}
});
}
}
@@ -7,7 +7,6 @@
import { Component, OnInit, inject, signal, computed } from '@angular/core';
import { CommonModule } from '@angular/common';
import { FormsModule } from '@angular/forms';
import { RouterLink } from '@angular/router';
import { MusicService } from '../../services/music.service';
import { AuthService } from '../../services/auth.service';
import { CommentsService } from '../../services/comments.service';
@@ -20,13 +19,9 @@ import { Music, MusicStatus, MusicType, CreateMusicDto, UpdateMusicDto, Comment,
@Component({
selector: 'app-music-list',
standalone: true,
imports: [CommonModule, FormsModule, RouterLink, PaginationComponent, LikeButtonComponent],
imports: [CommonModule, FormsModule, PaginationComponent, LikeButtonComponent],
template: `
<div class="container">
<div class="page-hero">
<img src="/assets/avatars/music-avatar.jpg" alt="Music avatar" class="page-avatar" />
</div>
<div class="header-section">
<h2>My Music Collection</h2>
@if (authService.isAdmin()) {
@@ -82,30 +77,9 @@ import { Music, MusicStatus, MusicType, CreateMusicDto, UpdateMusicDto, Comment,
<option [value]="MusicStatus.listening">Currently Listening</option>
<option [value]="MusicStatus.completed">Completed</option>
<option [value]="MusicStatus.wantToListen">Want to Listen</option>
<option [value]="MusicStatus.retired">Retired</option>
</select>
</div>
<div class="form-group">
<label for="dateStarted">Date Started</label>
<input
type="date"
id="dateStarted"
[(ngModel)]="newMusic.dateStarted"
name="dateStarted"
>
</div>
<div class="form-group">
<label for="dateFinished">Date Finished</label>
<input
type="date"
id="dateFinished"
[(ngModel)]="newMusic.dateFinished"
name="dateFinished"
>
</div>
<div class="form-group">
<label for="rating">Rating (1-10)</label>
<input
@@ -118,34 +92,6 @@ import { Music, MusicStatus, MusicType, CreateMusicDto, UpdateMusicDto, Comment,
>
</div>
<div class="form-row">
<div class="form-group">
<label for="timeHours">Time Spent (Hours)</label>
<input
type="number"
id="timeHours"
[(ngModel)]="newMusicTimeHours"
name="timeHours"
min="0"
placeholder="0"
(ngModelChange)="updateNewMusicTimeSpent()"
>
</div>
<div class="form-group">
<label for="timeMinutes">Time Spent (Minutes)</label>
<input
type="number"
id="timeMinutes"
[(ngModel)]="newMusicTimeMinutes"
name="timeMinutes"
min="0"
max="59"
placeholder="0"
(ngModelChange)="updateNewMusicTimeSpent()"
>
</div>
</div>
<div class="form-group">
<label for="notes">Notes</label>
<textarea
@@ -178,7 +124,8 @@ import { Music, MusicStatus, MusicType, CreateMusicDto, UpdateMusicDto, Comment,
</div>
<div class="form-group">
<div class="tags-input-container" aria-label="Tags">
<label>Tags</label>
<div class="tags-input-container">
@for (tag of newMusic.tags; track tag; let i = $index) {
<span class="tag">
{{ tag }}
@@ -195,7 +142,8 @@ import { Music, MusicStatus, MusicType, CreateMusicDto, UpdateMusicDto, Comment,
</div>
</div>
<div class="form-group" aria-label="External Links">
<div class="form-group">
<label>External Links</label>
<div class="links-list">
@for (link of newMusic.links; track link.url; let i = $index) {
<div class="link-item">
@@ -270,30 +218,9 @@ import { Music, MusicStatus, MusicType, CreateMusicDto, UpdateMusicDto, Comment,
<option [value]="MusicStatus.listening">Currently Listening</option>
<option [value]="MusicStatus.completed">Completed</option>
<option [value]="MusicStatus.wantToListen">Want to Listen</option>
<option [value]="MusicStatus.retired">Retired</option>
</select>
</div>
<div class="form-group">
<label for="edit-dateStarted">Date Started</label>
<input
type="date"
id="edit-dateStarted"
[(ngModel)]="editMusicData.dateStarted"
name="dateStarted"
>
</div>
<div class="form-group">
<label for="edit-dateFinished">Date Finished</label>
<input
type="date"
id="edit-dateFinished"
[(ngModel)]="editMusicData.dateFinished"
name="dateFinished"
>
</div>
<div class="form-group">
<label for="edit-rating">Rating (1-10)</label>
<input
@@ -306,34 +233,6 @@ import { Music, MusicStatus, MusicType, CreateMusicDto, UpdateMusicDto, Comment,
>
</div>
<div class="form-row">
<div class="form-group">
<label for="edit-timeHours">Time Spent (Hours)</label>
<input
type="number"
id="edit-timeHours"
[(ngModel)]="editMusicTimeHours"
name="timeHours"
min="0"
placeholder="0"
(ngModelChange)="updateEditMusicTimeSpent()"
>
</div>
<div class="form-group">
<label for="edit-timeMinutes">Time Spent (Minutes)</label>
<input
type="number"
id="edit-timeMinutes"
[(ngModel)]="editMusicTimeMinutes"
name="timeMinutes"
min="0"
max="59"
placeholder="0"
(ngModelChange)="updateEditMusicTimeSpent()"
>
</div>
</div>
<div class="form-group">
<label for="edit-notes">Notes</label>
<textarea
@@ -366,7 +265,8 @@ import { Music, MusicStatus, MusicType, CreateMusicDto, UpdateMusicDto, Comment,
</div>
<div class="form-group">
<div class="tags-input-container" aria-label="Tags">
<label>Tags</label>
<div class="tags-input-container">
@for (tag of editMusicData.tags; track tag; let i = $index) {
<span class="tag">
{{ tag }}
@@ -383,7 +283,8 @@ import { Music, MusicStatus, MusicType, CreateMusicDto, UpdateMusicDto, Comment,
</div>
</div>
<div class="form-group" aria-label="External Links">
<div class="form-group">
<label>External Links</label>
<div class="links-list">
@for (link of editMusicData.links; track link.url; let i = $index) {
<div class="link-item">
@@ -599,13 +500,6 @@ import { Music, MusicStatus, MusicType, CreateMusicDto, UpdateMusicDto, Comment,
>
Want to Listen ({{ wantToListenCount() }})
</button>
<button
(click)="setStatusFilter(MusicStatus.retired)"
[class.active]="statusFilter() === MusicStatus.retired"
class="filter-btn"
>
Retired ({{ retiredCount() }})
</button>
</div>
</div>
@@ -627,42 +521,74 @@ import { Music, MusicStatus, MusicType, CreateMusicDto, UpdateMusicDto, Comment,
<div class="music-grid">
@for (music of paginatedMusic(); track music.id) {
<div class="music-card" [class.completed]="music.status === MusicStatus.completed">
<a [routerLink]="['/music', music.id]" class="card-link">
<img [src]="music.coverArt || '/assets/default-cover.jpg'" [alt]="music.title" class="music-cover">
<div class="music-info">
<h3>{{ music.title }}</h3>
<p class="artist">{{ music.artist }}</p>
@if (music.type) {
<div class="badges">
<span class="type-badge type-{{ music.type }}">
{{ getTypeLabel(music.type) }}
</span>
<span class="status status-{{ music.status }}">
{{ getStatusLabel(music.status) }}
</span>
</div>
}
@if (music.rating) {
<div class="rating">
@for (star of [1,2,3,4,5,6,7,8,9,10]; track star) {
<span [class.filled]="star <= music.rating!">★</span>
}
</div>
@if (music.coverArt) {
<img [src]="music.coverArt" [alt]="music.title" class="music-cover">
} @else {
<div class="music-cover placeholder">
@switch (music.type) {
@case (MusicType.album) { 💿 }
@case (MusicType.single) { 🎵 }
@case (MusicType.ep) { 🎶 }
}
</div>
</a>
}
<div class="music-info">
<h3>{{ music.title }}</h3>
<p class="artist">{{ music.artist }}</p>
<div class="badges">
<span class="type-badge type-{{ music.type }}">
{{ getTypeLabel(music.type) }}
</span>
<span class="status status-{{ music.status }}">
{{ getStatusLabel(music.status) }}
</span>
</div>
@if (music.rating) {
<div class="rating">
@for (star of [1,2,3,4,5,6,7,8,9,10]; track star) {
<span [class.filled]="star <= music.rating!">★</span>
}
</div>
}
<div class="card-actions">
<app-like-button
entityType="music"
[entityId]="music.id"
></app-like-button>
@if (music.notes) {
<p class="notes">{{ music.notes }}</p>
}
@if (music.tags && music.tags.length > 0) {
<div class="tags-display">
@for (tag of music.tags; track tag) {
<span class="tag-chip">{{ tag }}</span>
}
</div>
}
@if (music.links && music.links.length > 0) {
<div class="links-display">
@for (link of music.links; track link.url) {
<a [href]="link.url" target="_blank" rel="noopener noreferrer" class="external-link">
{{ link.title }} ↗
</a>
}
</div>
}
@if (music.dateCompleted) {
<p class="date-completed">
Completed: {{ formatDate(music.dateCompleted) }}
</p>
}
@if (authService.isAdmin()) {
<div class="admin-actions">
<div class="actions">
<button (click)="startEdit(music)" class="btn btn-secondary btn-sm">
Edit
</button>
@@ -671,6 +597,85 @@ import { Music, MusicStatus, MusicType, CreateMusicDto, UpdateMusicDto, Comment,
</button>
</div>
}
<div class="comments-section">
<button (click)="toggleComments(music.id)" class="btn btn-secondary btn-sm comments-toggle">
{{ expandedComments()[music.id] ? 'Hide' : 'Show' }} Comments{{ comments()[music.id] ? ' (' + getCommentCount(music.id) + ')' : '' }}
</button>
@if (expandedComments()[music.id]) {
<div class="comments-container">
@if (authService.isAuthenticated()) {
@if (authService.user()?.isBanned) {
<div class="banned-notice">
You have been banned from commenting.
</div>
} @else {
<form (ngSubmit)="addComment(music.id)" class="comment-form">
<textarea
[(ngModel)]="newCommentContent[music.id]"
name="comment"
placeholder="Add a comment (Markdown supported)..."
rows="2"
></textarea>
<button type="submit" class="btn btn-primary btn-sm">Post Comment</button>
</form>
}
}
@if (commentsLoading()[music.id]) {
<div class="comments-loading">Loading comments...</div>
} @else {
@for (comment of comments()[music.id] || []; track comment.id) {
<div class="comment">
<div class="comment-header">
@if (comment.user.avatar) {
<img [src]="comment.user.avatar" [alt]="comment.user.username" class="comment-avatar">
}
<span class="comment-author">{{ comment.user.username }}</span>
@if (comment.user.inDiscord) {
<span class="discord-badge">Discord</span>
}
@if (comment.user.isVip) {
<span class="vip-badge">VIP</span>
}
@if (comment.user.isMod) {
<span class="mod-badge">Mod</span>
}
@if (comment.user.isStaff) {
<span class="staff-badge">Staff</span>
}
<span class="comment-date">{{ formatDate(comment.createdAt) }}</span>
@if (canEditComment(comment)) {
<button (click)="startEditComment(music.id, comment)" class="btn btn-secondary btn-xs">Edit</button>
}
@if (canDeleteComment(comment)) {
<button (click)="deleteComment(music.id, comment.id)" class="btn btn-danger btn-xs">Delete</button>
}
</div>
@if (editingCommentId() === comment.id) {
<div class="comment-edit-form">
<textarea
[(ngModel)]="editCommentContent"
name="editComment"
rows="3"
></textarea>
<div class="comment-edit-actions">
<button (click)="saveCommentEdit(music.id, comment.id)" class="btn btn-primary btn-xs">Save</button>
<button (click)="cancelCommentEdit()" class="btn btn-secondary btn-xs">Cancel</button>
</div>
</div>
} @else {
<div class="comment-content" [innerHTML]="sanitizeService.sanitizeHtml(comment.content)"></div>
}
</div>
} @empty {
<div class="no-comments">No comments yet. Be the first to comment!</div>
}
}
</div>
}
</div>
</div>
</div>
}
@@ -693,26 +698,6 @@ import { Music, MusicStatus, MusicType, CreateMusicDto, UpdateMusicDto, Comment,
padding: 2rem;
}
.page-hero {
text-align: center;
margin-bottom: 2rem;
}
.page-avatar {
width: 120px;
height: 120px;
border-radius: 50%;
object-fit: cover;
border: 3px solid #74b9ff;
box-shadow: 0 4px 12px rgba(116, 185, 255, 0.3);
transition: all 0.3s;
}
.page-avatar:hover {
transform: scale(1.05);
box-shadow: 0 8px 16px rgba(116, 185, 255, 0.5);
}
.header-section {
display: flex;
justify-content: space-between;
@@ -776,13 +761,6 @@ import { Music, MusicStatus, MusicType, CreateMusicDto, UpdateMusicDto, Comment,
box-shadow: 0 0 0 3px rgba(168, 87, 126, 0.2);
}
.form-row {
display: grid;
grid-template-columns: 1fr 1fr;
gap: 1rem;
margin-bottom: 1rem;
}
.form-actions {
display: flex;
gap: 1rem;
@@ -930,8 +908,6 @@ import { Music, MusicStatus, MusicType, CreateMusicDto, UpdateMusicDto, Comment,
overflow: hidden;
transition: all 0.3s;
backdrop-filter: blur(10px);
display: flex;
flex-direction: column;
}
.music-card:hover {
@@ -945,17 +921,6 @@ import { Music, MusicStatus, MusicType, CreateMusicDto, UpdateMusicDto, Comment,
border-color: var(--witch-mauve);
}
.card-link {
text-decoration: none;
color: inherit;
display: block;
flex: 1;
}
.card-link:hover h3 {
color: var(--witch-rose);
}
.music-cover {
width: 100%;
height: 250px;
@@ -977,7 +942,6 @@ import { Music, MusicStatus, MusicType, CreateMusicDto, UpdateMusicDto, Comment,
.music-info h3 {
margin: 0 0 0.5rem 0;
font-size: 1.1rem;
transition: color 0.3s;
}
.artist {
@@ -992,19 +956,6 @@ import { Music, MusicStatus, MusicType, CreateMusicDto, UpdateMusicDto, Comment,
margin: 0.75rem 0;
}
.card-actions {
padding: 1rem;
border-top: 1px solid var(--witch-lavender);
display: flex;
flex-direction: column;
gap: 0.5rem;
}
.admin-actions {
display: flex;
gap: 0.5rem;
}
.type-badge {
display: inline-block;
padding: 0.25rem 0.5rem;
@@ -1060,6 +1011,22 @@ import { Music, MusicStatus, MusicType, CreateMusicDto, UpdateMusicDto, Comment,
color: var(--witch-rose);
}
.notes {
font-size: 0.9rem;
color: var(--witch-plum);
margin: 0.5rem 0;
}
.date-completed {
font-size: 0.85rem;
color: var(--witch-plum);
margin-top: 0.5rem;
}
.actions {
margin-top: 1rem;
}
.btn {
padding: 0.5rem 1rem;
border: none;
@@ -1110,6 +1077,159 @@ import { Music, MusicStatus, MusicType, CreateMusicDto, UpdateMusicDto, Comment,
.btn-xs { padding: 0.15rem 0.5rem; font-size: 0.75rem; }
.comments-section {
margin-top: 1rem;
border-top: 1px solid var(--witch-lavender);
padding-top: 1rem;
}
.comments-toggle {
width: 100%;
}
.comments-container {
margin-top: 1rem;
}
.comment-form {
margin-bottom: 1rem;
}
.comment-form textarea {
width: 100%;
padding: 0.5rem;
border: 2px solid var(--witch-lavender);
border-radius: 4px;
font-size: 0.9rem;
background-color: var(--witch-moon);
color: var(--witch-purple);
resize: vertical;
margin-bottom: 0.5rem;
}
.comment-form textarea:focus {
outline: none;
border-color: var(--witch-rose);
}
.comment {
background: var(--witch-moon);
border: 1px solid var(--witch-lavender);
border-radius: 4px;
padding: 0.75rem;
margin-bottom: 0.5rem;
}
.comment-header {
display: flex;
align-items: center;
gap: 0.5rem;
margin-bottom: 0.5rem;
flex-wrap: wrap;
}
.comment-avatar {
width: 24px;
height: 24px;
border-radius: 50%;
}
.comment-author {
font-weight: 500;
color: var(--witch-plum);
}
.discord-badge {
background: #5865f2;
color: white;
padding: 0.1rem 0.4rem;
border-radius: 3px;
font-size: 0.7rem;
font-weight: 500;
}
.vip-badge {
background: linear-gradient(135deg, #ffd700, #ffaa00);
color: #1a1a1a;
padding: 0.1rem 0.4rem;
border-radius: 3px;
font-size: 0.7rem;
font-weight: 600;
}
.mod-badge {
background: linear-gradient(135deg, #00b894, #00cec9);
color: white;
padding: 0.1rem 0.4rem;
border-radius: 3px;
font-size: 0.7rem;
font-weight: 600;
}
.staff-badge {
background: linear-gradient(135deg, #e84393, #fd79a8);
color: white;
padding: 0.1rem 0.4rem;
border-radius: 3px;
font-size: 0.7rem;
font-weight: 600;
}
.comment-date {
font-size: 0.75rem;
color: var(--witch-mauve);
}
.comment-content {
font-size: 0.9rem;
color: var(--witch-purple);
}
.comments-loading,
.no-comments {
text-align: center;
padding: 1rem;
color: var(--witch-mauve);
font-size: 0.9rem;
}
.banned-notice {
background: #fef2f2;
border: 1px solid #fecaca;
color: #991b1b;
padding: 0.75rem 1rem;
border-radius: 4px;
text-align: center;
margin-bottom: 1rem;
font-size: 0.9rem;
}
.comment-edit-form {
margin-top: 0.5rem;
}
.comment-edit-form textarea {
width: 100%;
padding: 0.5rem;
border: 2px solid var(--witch-lavender);
border-radius: 4px;
font-size: 0.9rem;
background-color: var(--witch-moon);
color: var(--witch-purple);
resize: vertical;
}
.comment-edit-form textarea:focus {
outline: none;
border-color: var(--witch-rose);
}
.comment-edit-actions {
display: flex;
gap: 0.5rem;
margin-top: 0.5rem;
}
.image-preview {
margin-top: 0.5rem;
display: flex;
@@ -1191,6 +1311,21 @@ import { Music, MusicStatus, MusicType, CreateMusicDto, UpdateMusicDto, Comment,
opacity: 0.8;
}
.tags-display {
display: flex;
flex-wrap: wrap;
gap: 0.5rem;
margin: 0.5rem 0;
}
.tag-chip {
background: rgba(116, 185, 255, 0.2);
color: #74b9ff;
padding: 0.2rem 0.5rem;
border-radius: 4px;
font-size: 0.8rem;
}
.links-list {
margin-bottom: 0.5rem;
}
@@ -1215,6 +1350,28 @@ import { Music, MusicStatus, MusicType, CreateMusicDto, UpdateMusicDto, Comment,
flex: 1;
min-width: 120px;
}
.links-display {
display: flex;
flex-wrap: wrap;
gap: 0.5rem;
margin: 0.5rem 0;
}
.external-link {
color: #74b9ff;
text-decoration: none;
font-size: 0.85rem;
padding: 0.2rem 0.5rem;
background: rgba(116, 185, 255, 0.1);
border-radius: 4px;
transition: background 0.2s;
}
.external-link:hover {
background: rgba(116, 185, 255, 0.2);
text-decoration: underline;
}
`]
})
export class MusicListComponent implements OnInit {
@@ -1278,7 +1435,6 @@ export class MusicListComponent implements OnInit {
listeningCount = computed(() => this.music().filter(m => m.status === MusicStatus.listening).length);
completedCount = computed(() => this.music().filter(m => m.status === MusicStatus.completed).length);
wantToListenCount = computed(() => this.music().filter(m => m.status === MusicStatus.wantToListen).length);
retiredCount = computed(() => this.music().filter(m => m.status === MusicStatus.retired).length);
allTags = computed(() => {
const tagsSet = new Set<string>();
@@ -1332,13 +1488,11 @@ export class MusicListComponent implements OnInit {
totalFilteredMusic = computed(() => this.filteredMusic().length);
newMusic: Partial<CreateMusicDto> & { dateStarted?: Date; dateFinished?: Date } = {
newMusic: Partial<CreateMusicDto> = {
title: '',
artist: '',
type: MusicType.album,
status: MusicStatus.wantToListen,
dateStarted: undefined,
dateFinished: undefined,
rating: undefined,
notes: '',
tags: [],
@@ -1347,12 +1501,6 @@ export class MusicListComponent implements OnInit {
editMusicData: Partial<UpdateMusicDto> = {};
// Time tracking state
newMusicTimeHours = 0;
newMusicTimeMinutes = 0;
editMusicTimeHours = 0;
editMusicTimeMinutes = 0;
// Tags and links input state
newTagInput = '';
editTagInput = '';
@@ -1435,7 +1583,6 @@ export class MusicListComponent implements OnInit {
case MusicStatus.listening: return 'Currently Listening';
case MusicStatus.completed: return 'Completed';
case MusicStatus.wantToListen: return 'Want to Listen';
case MusicStatus.retired: return 'Retired';
}
}
@@ -1452,16 +1599,12 @@ export class MusicListComponent implements OnInit {
artist: '',
type: MusicType.album,
status: MusicStatus.wantToListen,
dateStarted: undefined,
dateFinished: undefined,
rating: undefined,
notes: '',
coverArt: undefined,
tags: [],
links: []
};
this.newMusicTimeHours = 0;
this.newMusicTimeMinutes = 0;
this.newMusicImagePreview.set(null);
this.imageError.set(null);
this.newTagInput = '';
@@ -1469,16 +1612,6 @@ export class MusicListComponent implements OnInit {
this.newLinkUrl = '';
}
updateNewMusicTimeSpent() {
const totalMinutes = (this.newMusicTimeHours * 60) + this.newMusicTimeMinutes;
this.newMusic.timeSpent = totalMinutes > 0 ? totalMinutes : undefined;
}
updateEditMusicTimeSpent() {
const totalMinutes = (this.editMusicTimeHours * 60) + this.editMusicTimeMinutes;
this.editMusicData.timeSpent = totalMinutes > 0 ? totalMinutes : undefined;
}
addTag(target: 'new' | 'edit') {
const input = target === 'new' ? this.newTagInput.trim() : this.editTagInput.trim();
if (!input) return;
@@ -1532,8 +1665,6 @@ export class MusicListComponent implements OnInit {
artist: this.newMusic.artist,
type: this.newMusic.type,
status: this.newMusic.status,
dateStarted: this.newMusic.dateStarted ? new Date(this.newMusic.dateStarted) : undefined,
dateFinished: this.newMusic.dateFinished ? new Date(this.newMusic.dateFinished) : undefined,
rating: this.newMusic.rating,
notes: this.newMusic.notes,
coverArt: this.newMusic.coverArt,
@@ -1562,32 +1693,18 @@ export class MusicListComponent implements OnInit {
artist: music.artist,
type: music.type,
status: music.status,
dateStarted: music.dateStarted,
dateFinished: music.dateFinished,
rating: music.rating,
notes: music.notes,
coverArt: music.coverArt,
tags: [...(music.tags || [])],
links: [...(music.links || [])],
timeSpent: music.timeSpent
links: [...(music.links || [])]
};
// Populate time fields from existing timeSpent
if (music.timeSpent) {
this.editMusicTimeHours = Math.floor(music.timeSpent / 60);
this.editMusicTimeMinutes = music.timeSpent % 60;
} else {
this.editMusicTimeHours = 0;
this.editMusicTimeMinutes = 0;
}
this.editMusicImagePreview.set(music.coverArt || null);
this.showAddForm.set(false);
this.imageError.set(null);
this.editTagInput = '';
this.editLinkTitle = '';
this.editLinkUrl = '';
// Scroll to top so the form is visible
window.scrollTo({ top: 0, behavior: 'smooth' });
}
cancelEdit() {
@@ -1604,13 +1721,7 @@ export class MusicListComponent implements OnInit {
const music = this.editingMusic();
if (!music || !this.editMusicData.title || !this.editMusicData.artist || !this.editMusicData.type || !this.editMusicData.status) return;
const updateData = {
...this.editMusicData,
dateStarted: this.editMusicData.dateStarted ? new Date(this.editMusicData.dateStarted) : undefined,
dateFinished: this.editMusicData.dateFinished ? new Date(this.editMusicData.dateFinished) : undefined,
};
this.musicService.updateMusic(music.id, updateData).subscribe(() => {
this.musicService.updateMusic(music.id, this.editMusicData).subscribe(() => {
this.loadMusic();
this.cancelEdit();
});
@@ -1620,19 +1731,6 @@ export class MusicListComponent implements OnInit {
return new Date(date).toLocaleDateString();
}
formatTimeSpent(minutes: number): string {
const hours = Math.floor(minutes / 60);
const mins = minutes % 60;
if (hours === 0) {
return `${mins}m`;
} else if (mins === 0) {
return `${hours}h`;
} else {
return `${hours}h ${mins}m`;
}
}
// Image handling methods
onImageSelected(event: Event, target: 'new' | 'edit' | 'suggest') {
const input = event.target as HTMLInputElement;
@@ -1821,7 +1919,7 @@ export class MusicListComponent implements OnInit {
try {
await this.suggestionService.createSuggestion({
entityType: SuggestionEntity.music,
entityType: SuggestionEntity.MUSIC,
title: this.suggestedMusic.title,
artist: this.suggestedMusic.artist,
type: this.suggestedMusic.type,
@@ -1834,21 +1932,4 @@ export class MusicListComponent implements OnInit {
alert('Failed to submit suggestion. Please try again.');
}
}
handleCommentEdit(musicId: string, event: { commentId: string; content: string }) {
this.commentsService.updateCommentOnMusic(musicId, event.commentId, event.content).subscribe({
next: (updatedComment) => {
this.comments.set({
...this.comments(),
[musicId]: (this.comments()[musicId] || []).map(c =>
c.id === event.commentId ? updatedComment : c
)
});
}
});
}
getCommentsSignal(musicId: string) {
return signal(this.comments()[musicId] || []);
}
}

Some files were not shown because too many files have changed in this diff Show More