Files
web-temp/layers/components/layouts/Footer.vue

259 lines
10 KiB
Vue

<template>
<footer id="footer" ref="footerRef" class="bg-black">
<div
class="inner relative max-w-7xl mx-auto px-10 py-9 text-[12px] text-gray-400 md:px-4 md:py-9 md:text-[12px]"
>
<div class="menu-area">
<ul class="flex items-center flex-wrap md:gap-6">
<li
v-for="(footerMenuItem, index) in footerLinks"
:key="index"
class="text-sm md:text-[11px] md:tracking-[-0.5px] relative flex items-center"
>
<NuxtLink
:to="footerMenuItem.url"
:target="footerMenuItem.target"
:class="[
footerMenuItem.active === 'y' && 'text-[#e04600]',
index === 2 && 'text-[#fff]',
'hover:text-gray-600 transition-colors',
]"
>
{{ footerMenuItem.title }}
</NuxtLink>
</li>
<li class="relative">
<button class="hover:text-gray-600 transition-colors" @click="toggleAgeRating">
{{ footerAgeRating }}
</button>
<div v-if="showAgeRating" class="game-rating-card absolute bottom-6 left-1/2 -translate-x-1/2 bg-[#383838] rounded-lg w-[340px] mx-auto z-10">
<!-- 헤더 -->
<div class="px-6 py-4 rounded-t-lg flex justify-between items-center">
<h3 class="text-white text-base">{{ footerAgeRating }}</h3>
<button class="text-white hover:text-gray-300 transition-colors" @click="toggleAgeRating">
<svg class="w-5 h-5" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M6 18L18 6M6 6l12 12"></path>
</svg>
</button>
</div>
<!-- 등급 아이콘 그리드 -->
<div class="px-6 pt-2 pb-6">
<!-- 아이콘은 52x60 사이즈 갭은 4px 4개씩 그리드 레이아웃 -->
<div class="grid grid-cols-4 gap-[4px] mb-4 max-w-[220px] justify-start items-start">
<!-- 19 등급 -->
<div v-for="image in getGameRatingImage" :key="image" class="text-center">
<img :src="image" alt="게임이용등급안내" class="w-full h-full object-contain" />
</div>
<div v-for="image in getContentInfoImage" :key="image" class="text-center">
<img :src="image" alt="게임이용등급안내" class="w-full h-full object-contain" />
</div>
</div>
</div>
<!-- 정보 테이블 -->
<div class="px-6 py-6 rounded-b-lg bg-[#A31639]">
<div class="space-y-2">
<div class="flex flex-start border-b border-white/10 pb-2">
<span class="text-white text-sm flex-1">{{ footerAgeRatingInfo[0] }}</span>
<span class="text-white text-sm flex-1">{{ footerData.game_rating_info.title }}</span>
</div>
<div class="flex flex-start border-b border-white/10 pb-2">
<span class="text-white text-sm flex-1">{{ footerAgeRatingInfo[1] }}</span>
<span class="text-white text-sm flex-1">{{ footerData.game_rating_info.rating_grade }}</span>
</div>
<div class="flex flex-start border-b border-white/10 pb-2">
<span class="text-white text-sm flex-1">{{ footerAgeRatingInfo[2] }}</span>
<span class="text-white text-sm flex-1">{{ footerData.game_rating_info.reg_no }}</span>
</div>
<div class="flex flex-start border-b border-white/10 pb-2">
<span class="text-white text-sm flex-1">{{ footerAgeRatingInfo[3] }}</span>
<span class="text-white text-sm flex-1">{{ footerData.game_rating_info.prod_date }}</span>
</div>
<div class="flex flex-start">
<span class="text-white text-sm flex-1">{{ footerAgeRatingInfo[4] }}</span>
<span class="text-white text-sm flex-1">{{ footerData.game_rating_info.rating_class_no }}</span>
</div>
</div>
</div>
</div>
</li>
</ul>
</div>
<div class="address-area mt-6">
<address class="not-italic text-gray-500">
<div class="row my-1.5 leading-5">
<span
v-dompurify-html="footerAddress"
class="[&_a]:cursor-pointer [&_a]:text-blue-500 [&_a]:underline"
></span>
</div>
</address>
</div>
<div class="mt-6 md:mt-6">
<div class="text-xs text-white/30">{{ footerCaution }}</div>
</div>
<div class="copyright-area mt-6 text-gray-500 md:mt-4">
<span>&copy; Smilegate. All rights reserved</span>
</div>
<div class="logo-area flex mt-6 md:mt-6">
<a
href="https://www.smilegate.com"
target="_blank"
class="smilegate"
>
<img
src="https://static-pubcomm.gate8.com/local/template/l9/common/logo_smilegate.png"
alt="스마일게이트 로고"
class="w-auto h-auto"
/>
</a>
<a
v-if="setDevCi.dev_ci_yn"
:href="footerData.use_dev_ci_url ? setDevCi.dev_ci_img_path : '#'"
target="_blank"
class="nx3 ml-2.5 md:ml-4"
>
<img
:src="`${staticUrl}${setDevCi.dev_ci_img_path}`"
alt="CI"
class="w-auto h-[24px]"
/>
</a>
</div>
<div
class="language-area absolute bottom-7 right-10 text-white md:bottom-5.5 md:right-4"
>
<BlocksLanguageSwitcher />
<!-- <SelectLanguage /> -->
<!-- <AtomsLanguageSwitcher /> -->
</div>
</div>
</footer>
</template>
<script setup lang="ts">
import type { FooterMenuItem, FooterData, DevCiConfig } from '#layers/types/Common'
const config = useRuntimeConfig()
const staticUrl = config.public.staticUrl
const runType = config.public.runType
const translationApi = `${runType}/test/multilingual/test_common_template.json`
const result = await useApiData({ baseApiUrl: staticUrl, url: translationApi })
const { tm } = useI18n({
useScope: 'local',
messages: result
})
// const { tm } = useI18n()
const gameDataStore = useGameDataStore()
const { gameData } = storeToRefs(gameDataStore)
// const path = ref<string>(`${staticUrl}/local/template/${gameData.value.s3_folder_name}`)
// 공통다국어 data
const footerLinks = computed((): FooterMenuItem[] => {
const menu = (tm as any)('Footer_Menu')
return Array.isArray(menu) ? menu as FooterMenuItem[] : []
})
const footerData = ref(gameData.value?.footer_json as unknown as FooterData)
const setDevCi = ref<DevCiConfig>({
dev_ci_yn: gameData.value?.footer_dev_ci_img_yn as boolean,
dev_ci_img_path: gameData.value?.footer_dev_ci_img_path as string,
})
///local/template/common/grades_age
const getGameRatingImage = computed((): string[] => {
const contentInfo = footerData.value.game_rating_info.rating_type.split(',')
// rating_type 12, 15, 18, 19 에 따라 이미지명을 가져오고 이미지를 반환
return contentInfo.map(item => {
switch (item) {
case '12':
return getImageHost('/images/common/grades_age/Type12.svg', { imageType: 'common' })
case '15':
return getImageHost('/images/common/grades_age/Type15.svg', { imageType: 'common' })
case '19':
return getImageHost('/images/common/grades_age/Type19.svg', { imageType: 'common' })
case 'all':
return getImageHost('/images/common/grades_age/TypeAll.svg', { imageType: 'common' })
case 'e':
return getImageHost('/images/common/grades_age/TypeExempt.svg', { imageType: 'common' })
default:
return getImageHost('/images/common/grades_age/TypeTest.svg', { imageType: 'common' })
}
})
})
const getContentInfoImage = computed((): string[] => {
const contentInfo = footerData.value.game_rating_info.content_info.split(',')
contentInfo.pop()
return contentInfo.map(item => {
switch (item) {
case '1':
return getImageHost('/images/common/grades_use/Type-sexual.svg', { imageType: 'common' })
case '2':
return getImageHost('/images/common/grades_use/Type-fear.svg', { imageType: 'common' })
case '3':
return getImageHost('/images/common/grades_use/Type-inapposite.svg', { imageType: 'common' })
case '4':
return getImageHost('/images/common/grades_use/Type-drug.svg', { imageType: 'common' })
case '5':
return getImageHost('/images/common/grades_use/Type-crime.svg', { imageType: 'common' })
case '6':
return getImageHost('/images/common/grades_use/Type-speculation.svg', { imageType: 'common' })
case '7':
return getImageHost('/images/common/grades_use/Type-violence.svg', { imageType: 'common' })
}
})
})
const showAgeRating = ref<boolean>(false)
const toggleAgeRating = (): void => {
showAgeRating.value = !showAgeRating.value
}
// 템플릿에서 사용할 다국어 텍스트들
const footerAgeRating = computed((): string => {
const text = (tm as any)('Footer_AgeRating')
return typeof text === 'string' ? text : ''
})
const footerAgeRatingInfo = computed((): string[] => {
const info = (tm as any)('Footer_AgeRating_Info')
return Array.isArray(info) ? info : []
})
const footerAddress = computed((): string => {
const address = (tm as any)('Footer_Address')
return typeof address === 'string' ? address : ''
})
const footerCaution = computed((): string => {
const caution = (tm as any)('Footer_caution')
return typeof caution === 'string' ? caution : ''
})
</script>
<style scoped>
/* 태국어 폰트 크기 조정 */
@media (max-width: 411px) {
:global(.lang-th) .menu-area li {
font-size: 10px;
}
}
@media (max-width: 321px) {
:global(.lang-th) .menu-area li {
font-size: 9px;
}
}
</style>