Files
MediaCollectorLibary/resources/views/games/show.twig
Lars Behrends 04140786a7 Stuff i guess ?
2025-10-31 00:24:17 +01:00

502 lines
30 KiB
Twig

{% extends "layouts/app.twig" %}
{% block content %}
<div class="container py-4">
<!-- Game Header -->
<div class="card mb-4 shadow-sm">
<div class="card-header bg-white">
<div class="d-flex justify-content-between align-items-center">
<div class="d-flex align-items-center">
{% if main_game.image_url %}
<img class="me-3 rounded" src="{{ main_game.image_url }}" alt="{{ main_game.title }}" style="width: 64px; height: 64px; object-fit: cover;">
{% else %}
<div class="bg-light rounded d-flex align-items-center justify-content-center me-3" style="width: 64px; height: 64px;">
<i class="bi bi-controller text-muted" style="font-size: 1.5rem;"></i>
</div>
{% endif %}
<div>
<h1 class="h4 mb-1">{{ main_game.title }}</h1>
<div class="d-flex align-items-center gap-3">
<span class="badge bg-primary">
{{ platform_versions|length }} platform{{ platform_versions|length > 1 ? 's' : '' }}
</span>
{% if main_game.genre %}
<span class="text-muted small">{{ main_game.genre }}</span>
{% endif %}
</div>
</div>
</div>
<a href="{{ path_for('games.index') }}" class="btn btn-outline-primary btn-sm">
<i class="bi bi-arrow-left me-1"></i> Back to Games
</a>
</div>
</div>
<!-- Platform Tabs -->
<div class="card-body p-0">
<ul class="nav nav-tabs px-3" id="platformTabs" role="tablist">
{% for version in platform_versions %}
<li class="nav-item" role="presentation">
<button
class="nav-link {{ loop.first ? 'active' : 'text-muted' }} platform-tab"
id="tab-{{ version.platform|lower }}-{{ version.source_id }}"
data-bs-toggle="tab"
data-bs-target="#content-{{ version.platform|lower }}-{{ version.source_id }}"
type="button"
role="tab"
data-platform="{{ version.platform }}"
data-source="{{ version.source_id }}"
>
{{ version.platform }}
{% if version.source_name %}
<small class="text-muted">({{ version.source_name }})</small>
{% endif %}
</button>
</li>
{% endfor %}
</ul>
<!-- Tab Content -->
<div class="tab-content p-4" id="platformTabsContent">
{% for version in platform_versions %}
<div
class="tab-pane fade {{ loop.first ? 'show active' }}"
id="content-{{ version.platform|lower }}-{{ version.source_id }}"
role="tabpanel"
data-platform="{{ version.platform }}"
data-source="{{ version.source_id }}"
>
<div class="row g-4">
<!-- Game Info -->
<div class="col-md-6 pb-3">
<div class="card h-100">
<div class="card-header">
<h5 class="card-title mb-0">Game Information</h5>
</div>
<div class="card-body">
<dl class="row mb-0">
{% if version.developer %}
<dt class="col-sm-5 text-muted small">Developer</dt>
<dd class="col-sm-7">{{ version.developer }}</dd>
{% endif %}
{% if version.publisher %}
<dt class="col-sm-5 text-muted small">Publisher</dt>
<dd class="col-sm-7">{{ version.publisher }}</dd>
{% endif %}
{% if version.release_date %}
<dt class="col-sm-5 text-muted small">Release Date</dt>
<dd class="col-sm-7">{{ version.release_date|date('M j, Y') }}</dd>
{% endif %}
<dt class="col-sm-5 text-muted small">Playtime</dt>
<dd class="col-sm-7">{{ version.playtime_minutes|format_duration }}</dd>
{% if version.rating %}
<dt class="col-sm-5 text-muted small">Rating</dt>
<dd class="col-sm-7">
<div class="d-flex align-items-center">
<div class="progress flex-grow-1 me-2" style="height: 6px;">
<div class="progress-bar bg-warning" role="progressbar" style="width: {{ version.rating * 10 }}%" aria-valuenow="{{ version.rating * 10 }}" aria-valuemin="0" aria-valuemax="100"></div>
</div>
<span class="small">{{ version.rating }}/10</span>
</div>
</dd>
{% endif %}
{% if version.completion_percentage > 0 %}
<dt class="col-sm-5 text-muted small">Completion</dt>
<dd class="col-sm-7">
<div class="d-flex align-items-center">
<div class="progress flex-grow-1 me-2" style="height: 6px;">
<div class="progress-bar bg-success" role="progressbar" style="width: {{ version.completion_percentage }}%" aria-valuenow="{{ version.completion_percentage }}" aria-valuemin="0" aria-valuemax="100"></div>
</div>
<span class="small">{{ version.completion_percentage }}%</span>
</div>
</dd>
{% endif %}
</dl>
</div>
</div>
</div>
<!-- Platform Stats -->
<div class="col-md-6 pb-3">
<div class="card h-100">
<div class="card-header">
<h5 class="card-title mb-0">Platform Statistics</h5>
</div>
<div class="card-body">
<dl class="row mb-0">
<dt class="col-sm-5 text-muted small">Source</dt>
<dd class="col-sm-7">{{ version.source_name }}</dd>
{% if version.last_played_at %}
<dt class="col-sm-5 text-muted small">Last Played</dt>
<dd class="col-sm-7">
<span data-bs-toggle="tooltip" data-bs-placement="top" title="{{ version.last_played_at|date('F j, Y H:i') }}">
{{ version.last_played_at|date('M j, Y') }}
</span>
</dd>
{% endif %}
{% if version.is_installed %}
<dt class="col-sm-5 text-muted small">Status</dt>
<dd class="col-sm-7">
<span class="badge bg-success">
<i class="bi bi-check-circle me-1"></i> Installed
</span>
</dd>
{% endif %}
{% if version.is_favorite %}
<dt class="col-sm-5 text-muted small">Favorite</dt>
<dd class="col-sm-7">
<span class="badge bg-danger">
<i class="bi bi-heart-fill me-1"></i> Yes
</span>
</dd>
{% endif %}
</dl>
<!-- Platform-specific metadata -->
{% set metadata = version.metadata|json_decode %}
{% if metadata %}
<hr class="my-3">
<h6 class="fw-bold mb-2">Platform Details</h6>
<dl class="row mb-0">
{% if metadata.appid %}
<dt class="col-sm-5 text-muted small">App ID</dt>
<dd class="col-sm-7">
<code>{{ metadata.appid }}</code>
</dd>
{% endif %}
{% if metadata.playtime_windows or metadata.playtime_mac or metadata.playtime_linux %}
<dt class="col-sm-5 text-muted small">Platform Playtime</dt>
<dd class="col-sm-7">
<div class="d-flex flex-column gap-1">
{% if metadata.playtime_windows %}
<div class="d-flex align-items-center">
<i class="bi bi-windows me-2 text-primary"></i>
<span class="small">{{ metadata.playtime_windows|format_duration }}</span>
</div>
{% endif %}
{% if metadata.playtime_mac %}
<div class="d-flex align-items-center">
<i class="bi bi-apple me-2 text-muted"></i>
<span class="small">{{ metadata.playtime_mac|format_duration }}</span>
</div>
{% endif %}
{% if metadata.playtime_linux %}
<div class="d-flex align-items-center">
<i class="bi bi-ubuntu me-2 text-warning"></i>
<span class="small">{{ metadata.playtime_linux|format_duration }}</span>
</div>
{% endif %}
</div>
</dd>
{% endif %}
</dl>
{% endif %}
</div>
</div>
</div>
</div>
{% if version.description %}
<div class="col-12 pb-3">
<div class="card">
<div class="card-header">
<h5 class="card-title mb-0">Description</h5>
</div>
<div class="card-body">
<p class="card-text">{{ version.description }}</p>
</div>
</div>
</div>
{% endif %}
<!-- Playnite Rich Metadata -->
<div class="col-12">
<div class="row g-3">
{% set playniteGenres = version.genres_json|json_decode %}
{% if playniteGenres %}
<div class="col-12">
<div class="card h-100">
<div class="card-header">
<h5 class="card-title mb-0">Genres</h5>
</div>
<div class="card-body">
<div class="d-flex flex-wrap gap-2">
{% for genre in playniteGenres %}
<span class="badge bg-primary bg-opacity-10 text-primary">
<i class="bi bi-tag me-1"></i> {{ genre.Name }}
</span>
{% endfor %}
</div>
</div>
</div>
</div>
{% endif %}
{% set playniteTags = version.tags_json|json_decode %}
{% if playniteTags %}
<div class="col-12">
<div class="card h-100">
<div class="card-header">
<h5 class="card-title mb-0">Tags</h5>
</div>
<div class="card-body">
<div class="d-flex flex-wrap gap-2">
{% for tag in playniteTags|slice(0, 20) %}
<span class="badge bg-secondary bg-opacity-10 text-secondary">
<i class="bi bi-tag-fill me-1"></i> {{ tag.Name }}
</span>
{% endfor %}
{% if playniteTags|length > 20 %}
<span class="badge bg-light text-muted">+{{ playniteTags|length - 20 }} more</span>
{% endif %}
</div>
</div>
</div>
</div>
{% endif %}
{% set playniteFeatures = version.features_json|json_decode %}
{% if playniteFeatures %}
<div class="col-12">
<div class="card h-100">
<div class="card-header">
<h5 class="card-title mb-0">Features</h5>
</div>
<div class="card-body">
<div class="d-flex flex-wrap gap-2">
{% for feature in playniteFeatures %}
<span class="badge bg-success bg-opacity-10 text-success">
<i class="bi bi-check-circle me-1"></i> {{ feature.Name }}
</span>
{% endfor %}
</div>
</div>
</div>
</div>
{% endif %}
{% set playniteLinks = version.links_json|json_decode %}
{% if playniteLinks %}
<div class="col-12">
<div class="card">
<div class="card-header">
<h5 class="card-title mb-0">Links</h5>
</div>
<div class="card-body">
<div class="row g-2">
{% for link in playniteLinks %}
<div class="col-12 col-sm-6">
<a href="{{ link.Url }}" target="_blank" rel="noopener noreferrer" class="btn btn-outline-secondary w-100 text-start">
<i class="bi bi-box-arrow-up-right me-2"></i> {{ link.Name }}
</a>
</div>
{% endfor %}
</div>
</div>
</div>
</div>
{% endif %}
{% set playniteSeries = version.series_json|json_decode %}
{% if playniteSeries %}
<div class="col-12">
<div class="card h-100">
<div class="card-header">
<h5 class="card-title mb-0">Series</h5>
</div>
<div class="card-body">
<div class="d-flex flex-wrap gap-2">
{% for series in playniteSeries %}
<span class="badge bg-purple bg-opacity-10 text-purple">
<i class="bi bi-collection me-1"></i> {{ series.Name }}
</span>
{% endfor %}
</div>
</div>
</div>
</div>
{% endif %}
{% set playniteAgeRatings = version.age_ratings_json|json_decode %}
{% if playniteAgeRatings %}
<div class="col-12">
<div class="card h-100">
<div class="card-header">
<h5 class="card-title mb-0">Age Ratings</h5>
</div>
<div class="card-body">
<div class="d-flex flex-wrap gap-2">
{% for ageRating in playniteAgeRatings %}
<span class="badge bg-warning bg-opacity-10 text-warning">
<i class="bi bi-exclamation-triangle me-1"></i> {{ ageRating.Name }}
</span>
{% endfor %}
</div>
</div>
</div>
</div>
{% endif %}
<!-- Enhanced Ratings Display -->
{% if version.critic_score or version.community_score or version.user_score %}
<div class="col-12">
<div class="card">
<div class="card-header">
<h5 class="card-title mb-0">Ratings</h5>
</div>
<div class="card-body">
<div class="row g-4">
{% if version.critic_score %}
<div class="col-12 col-md-4">
<div class="d-flex align-items-center p-3 bg-light rounded">
<div class="flex-shrink-0 me-3">
<i class="bi bi-star-fill text-warning" style="font-size: 1.75rem;"></i>
</div>
<div>
<div class="text-muted small">Critic Score</div>
<div class="h3 mb-0 fw-bold">{{ version.critic_score }}%</div>
</div>
</div>
</div>
{% endif %}
{% if version.community_score %}
<div class="col-12 col-md-4">
<div class="d-flex align-items-center p-3 bg-light rounded">
<div class="flex-shrink-0 me-3">
<i class="bi bi-people-fill text-primary" style="font-size: 1.75rem;"></i>
</div>
<div>
<div class="text-muted small">Community Score</div>
<div class="h3 mb-0 fw-bold">{{ version.community_score }}%</div>
</div>
</div>
</div>
{% endif %}
{% if version.user_score %}
<div class="col-12 col-md-4">
<div class="d-flex align-items-center p-3 bg-light rounded">
<div class="flex-shrink-0 me-3">
<i class="bi bi-person-check-fill text-success" style="font-size: 1.75rem;"></i>
</div>
<div>
<div class="text-muted small">Your Score</div>
<div class="h3 mb-0 fw-bold">{{ version.user_score }}%</div>
</div>
</div>
</div>
{% endif %}
</div>
</div>
</div>
</div>
{% endif %}
<!-- Playnite-specific Statistics -->
{% if version.play_count or version.install_size or version.completion_status %}
<div class="col-12">
<div class="card">
<div class="card-header">
<h5 class="card-title mb-0">Playnite Statistics</h5>
</div>
<div class="card-body">
<div class="row g-4">
{% if version.play_count %}
<div class="col-12 col-md-4">
<div class="d-flex align-items-center p-3 bg-light rounded">
<div class="flex-shrink-0 me-3">
<i class="bi bi-play-circle-fill text-primary" style="font-size: 1.75rem;"></i>
</div>
<div>
<div class="text-muted small">Play Count</div>
<div class="h3 mb-0 fw-bold">{{ version.play_count }}</div>
</div>
</div>
</div>
{% endif %}
{% if version.install_size %}
<div class="col-12 col-md-4">
<div class="d-flex align-items-center p-3 bg-light rounded">
<div class="flex-shrink-0 me-3">
<i class="bi bi-hdd-fill text-secondary" style="font-size: 1.75rem;"></i>
</div>
<div>
<div class="text-muted small">Install Size</div>
<div class="h3 mb-0 fw-bold">{{ (version.install_size / 1024 / 1024 / 1024)|round(1) }} GB</div>
</div>
</div>
</div>
{% endif %}
{% if version.completion_status %}
<div class="col-12 col-md-4">
<div class="d-flex align-items-center p-3 bg-light rounded">
<div class="flex-shrink-0 me-3">
<i class="bi bi-check-circle-fill text-success" style="font-size: 1.75rem;"></i>
</div>
<div>
<div class="text-muted small">Completion</div>
<div class="h3 mb-0 fw-bold">{{ version.completion_status }}</div>
</div>
</div>
</div>
{% endif %}
</div>
</div>
</div>
</div>
{% endif %}
</div>
</div>
</div>
{% endfor %}
</div>
</div>
</div>
</div>
<script>
// Initialize Bootstrap tooltips
document.addEventListener('DOMContentLoaded', function() {
// Initialize all tooltips
var tooltipTriggerList = [].slice.call(document.querySelectorAll('[data-bs-toggle="tooltip"]'));
var tooltipList = tooltipTriggerList.map(function (tooltipTriggerEl) {
return new bootstrap.Tooltip(tooltipTriggerEl);
});
// Platform tab switching functionality (using Bootstrap's tab component)
const platformTabs = document.querySelectorAll('.platform-tab');
platformTabs.forEach(tab => {
tab.addEventListener('shown.bs.tab', function (event) {
const platform = event.target.dataset.platform;
const source = event.target.dataset.source;
// Update active tab styling
platformTabs.forEach(t => {
t.classList.remove('active', 'border-primary', 'text-primary');
t.classList.add('text-muted');
});
this.classList.remove('text-muted');
this.classList.add('active', 'border-primary', 'text-primary');
// Any additional tab change logic can go here
console.log(`Switched to platform: ${platform}, source: ${source}`);
});
});
});
</script>
{% endblock %}