mirror of
https://github.com/Spythere/pojazdownik.git
synced 2026-05-03 11:45:34 +00:00
224 lines
5.2 KiB
Vue
224 lines
5.2 KiB
Vue
<template>
|
|
<section class="train-image-section">
|
|
<div v-if="store.chosenVehicle">
|
|
<div class="image-wrapper">
|
|
<img
|
|
:src="getThumbnailURL(store.chosenVehicle.type, 'small')"
|
|
:data-preview-available="isDataPreviewAvailable"
|
|
:data-sponsor-only="store.chosenVehicle.sponsorOnlyTimestamp && store.chosenVehicle.sponsorOnlyTimestamp > Date.now()"
|
|
:data-team-only="store.chosenVehicle.teamOnly"
|
|
@click="onImageClick"
|
|
@keydown.enter="onImageClick"
|
|
@load="onImageLoad"
|
|
@error="onImageError"
|
|
tabindex="0"
|
|
/>
|
|
</div>
|
|
|
|
<div class="image-info">
|
|
<b class="text--accent">{{ store.chosenVehicle.type }}</b> •
|
|
<b style="color: #ccc">
|
|
{{ $t(`preview.${isTractionUnit(store.chosenVehicle) ? store.chosenVehicle.group : store.chosenVehicle.group}`) }}
|
|
</b>
|
|
|
|
<div style="color: #ccc">
|
|
<div>
|
|
{{ store.chosenVehicle.length }}m | {{ (store.chosenVehicle.weight / 1000).toFixed(1) }}t | {{ store.chosenVehicle.maxSpeed }} km/h
|
|
</div>
|
|
|
|
<div v-if="isTractionUnit(store.chosenVehicle)">{{ $t('preview.cabin') }} {{ store.chosenVehicle.cabinType }}</div>
|
|
|
|
<div v-else>
|
|
{{
|
|
store.chosenVehicle.group == 'wagon-freight'
|
|
? $t(`usage.${store.chosenVehicle.constructionType}`)
|
|
: `${$t('preview.construction')} ${store.chosenVehicle.constructionType}`
|
|
}}
|
|
</div>
|
|
|
|
<b v-if="store.chosenVehicle.sponsorOnlyTimestamp && store.chosenVehicle.sponsorOnlyTimestamp > Date.now()" class="sponsor-only">
|
|
{{
|
|
$t('preview.sponsor-only', [
|
|
new Date(store.chosenVehicle.sponsorOnlyTimestamp).toLocaleDateString($i18n.locale == 'pl' ? 'pl-PL' : 'en-GB'),
|
|
])
|
|
}}
|
|
</b>
|
|
|
|
<b v-if="store.chosenVehicle.teamOnly" class="team-only">{{ $t('preview.team-only') }}</b>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<div v-else>
|
|
<img src="/images/placeholder.jpg" alt="placeholder image" />
|
|
<div class="image-info">{{ $t('preview.desc') }}</div>
|
|
</div>
|
|
</section>
|
|
</template>
|
|
|
|
<script lang="ts">
|
|
import { computed, defineComponent } from 'vue';
|
|
import { useStore } from '../../store';
|
|
import { isTractionUnit } from '../../utils/vehicleUtils';
|
|
import imageMixin from '../../mixins/imageMixin';
|
|
import { watch } from 'vue';
|
|
import { storeToRefs } from 'pinia';
|
|
import { Ref } from 'vue';
|
|
import { ref } from 'vue';
|
|
|
|
export default defineComponent({
|
|
mixins: [imageMixin],
|
|
|
|
data() {
|
|
return {
|
|
noImageAvailable: false,
|
|
};
|
|
},
|
|
|
|
setup() {
|
|
const store = useStore();
|
|
const { chosenVehicle } = storeToRefs(store);
|
|
|
|
const imageStatus = ref('none') as Ref<'none' | 'loading' | 'loaded' | 'error'>;
|
|
|
|
watch(chosenVehicle, () => {
|
|
imageStatus.value = 'loading';
|
|
});
|
|
|
|
return {
|
|
store,
|
|
imageStatus,
|
|
chosenVehicle: computed(() => store.chosenVehicle),
|
|
};
|
|
},
|
|
|
|
computed: {
|
|
isDataPreviewAvailable() {
|
|
return this.imageStatus == 'loaded' && this.store.chosenVehicle != null;
|
|
},
|
|
},
|
|
|
|
methods: {
|
|
isTractionUnit,
|
|
|
|
onImageError(e: Event) {
|
|
const el = e.target as HTMLImageElement;
|
|
if (el.src == '/images/no-vehicle-image.png') return;
|
|
|
|
el.src = '/images/no-vehicle-image.png';
|
|
this.imageStatus = 'error';
|
|
},
|
|
|
|
onImageLoad(e: Event) {
|
|
if (this.imageStatus == 'error') return;
|
|
|
|
this.imageStatus = 'loaded';
|
|
},
|
|
|
|
onImageClick(e: Event) {
|
|
const target = e.target as HTMLImageElement;
|
|
|
|
const chosenVehicle = this.store.chosenVehicle;
|
|
|
|
if (!chosenVehicle) return;
|
|
|
|
this.store.lastFocusedElement = target;
|
|
|
|
if (this.isDataPreviewAvailable) {
|
|
this.store.vehiclePreviewSrc = this.getThumbnailURL(chosenVehicle.type, 'large');
|
|
}
|
|
},
|
|
},
|
|
});
|
|
</script>
|
|
|
|
<style lang="scss" scoped>
|
|
@use '@/styles/responsive';
|
|
|
|
.train-image-section {
|
|
display: flex;
|
|
flex-direction: column;
|
|
align-items: center;
|
|
text-align: center;
|
|
|
|
min-height: 250px;
|
|
|
|
& > div {
|
|
position: relative;
|
|
width: 100%;
|
|
max-width: 380px;
|
|
}
|
|
}
|
|
|
|
img {
|
|
width: 100%;
|
|
|
|
&[data-preview-available='true'] {
|
|
border: 1px solid white;
|
|
cursor: zoom-in;
|
|
}
|
|
|
|
&[data-sponsor-only='true'] {
|
|
border: 1px solid var(--accentColor);
|
|
}
|
|
|
|
&[data-team-only='true'] {
|
|
border: 1px solid var(--teamColor);
|
|
}
|
|
}
|
|
|
|
.image-wrapper {
|
|
position: relative;
|
|
min-height: 220px;
|
|
}
|
|
|
|
.loading-placeholder {
|
|
position: absolute;
|
|
top: 0;
|
|
left: 0;
|
|
height: 100%;
|
|
width: 100%;
|
|
|
|
display: flex;
|
|
align-items: center;
|
|
justify-content: center;
|
|
|
|
background-color: #2b2b2b;
|
|
padding: 1em;
|
|
font-size: 1.2em;
|
|
}
|
|
|
|
.image-info {
|
|
font-size: 1.1em;
|
|
padding: 0.5em;
|
|
// margin: 0.5em auto;
|
|
line-height: 1.35;
|
|
|
|
width: 100%;
|
|
max-width: 380px;
|
|
|
|
background-color: var(--secondaryColor);
|
|
font-weight: bold;
|
|
}
|
|
|
|
.placeholder {
|
|
height: 250px;
|
|
|
|
background-color: var(--bgColor);
|
|
}
|
|
|
|
.sponsor-only {
|
|
color: var(--accentColor);
|
|
}
|
|
|
|
.team-only {
|
|
color: var(--teamColor);
|
|
}
|
|
|
|
@include responsive.midScreen {
|
|
.train-image-section {
|
|
justify-content: center;
|
|
}
|
|
}
|
|
</style>
|