<template>
|
<div class="sidebar-logo-container" :class="{ collapse }">
|
<transition name="sidebarLogoFade">
|
<router-link v-if="collapse" key="collapse" class="sidebar-logo-link" to="/">
|
<img :src="faviconUrl" class="sidebar-logo sidebar-favicon" alt="站点图标" />
|
</router-link>
|
<router-link v-else key="expand" class="sidebar-logo-link" :style="expandLogoLinkStyle" to="/">
|
<img v-if="logoUrl" :src="logoUrl" class="sidebar-logo" @error="handleImageError" alt="公司Logo" />
|
<h1 v-if="!logoUrl" class="sidebar-title">{{ title }}</h1>
|
</router-link>
|
</transition>
|
</div>
|
</template>
|
|
<script setup>
|
import { ref, computed, onMounted, watch } from 'vue'
|
import useUserStore from '@/store/modules/user'
|
import defaultLogo from '@/assets/logo/logo.png'
|
|
defineProps({
|
collapse: {
|
type: Boolean,
|
required: true
|
}
|
})
|
|
const title = import.meta.env.VITE_APP_TITLE
|
const userStore = useUserStore()
|
const baseUrl = import.meta.env.BASE_URL || '/'
|
const faviconUrl = `${baseUrl.replace(/\/?$/, '/') }favicon.ico`.replace(/([^:]\/)\/+/g, '$1')
|
|
const cleanFactoryName = computed(() => {
|
if (!userStore.currentFactoryName) return ''
|
return userStore.currentFactoryName.trim()
|
})
|
|
const logoUrl = ref('')
|
|
const expandLogoLinkStyle = computed(() => {
|
if (!logoUrl.value) {
|
return { '--logo-bg-image': 'none' }
|
}
|
const escaped = String(logoUrl.value).replace(/"/g, '\\"')
|
return { '--logo-bg-image': `url("${escaped}")` }
|
})
|
|
const updateLogoUrl = () => {
|
if (!cleanFactoryName.value) {
|
logoUrl.value = defaultLogo
|
return
|
}
|
|
try {
|
const dynamicLogo = import.meta.glob('/src/assets/logo/*.png', { eager: true })
|
const logoPath = `/src/assets/logo/${cleanFactoryName.value}.png`
|
|
if (dynamicLogo[logoPath]) {
|
logoUrl.value = dynamicLogo[logoPath].default
|
} else {
|
logoUrl.value = defaultLogo
|
}
|
} catch (error) {
|
console.error('加载工厂 Logo 失败:', error)
|
logoUrl.value = defaultLogo
|
}
|
}
|
|
onMounted(() => {
|
updateLogoUrl()
|
watch(() => userStore.currentFactoryName, updateLogoUrl)
|
})
|
|
const handleImageError = () => {
|
logoUrl.value = defaultLogo
|
}
|
</script>
|
|
<style lang="scss" scoped>
|
@import '@/assets/styles/variables.module.scss';
|
|
.sidebarLogoFade-enter-active {
|
transition: opacity 1.5s;
|
}
|
|
.sidebarLogoFade-enter,
|
.sidebarLogoFade-leave-to {
|
opacity: 0;
|
}
|
|
.sidebar-logo-container {
|
position: relative;
|
width: 100% !important;
|
height: 78px !important;
|
line-height: 78px;
|
background: transparent;
|
border-bottom: 1px solid rgba(255, 255, 255, 0.08);
|
text-align: left;
|
overflow: hidden;
|
box-shadow: none;
|
backdrop-filter: none;
|
transition: all 0.3s ease;
|
|
.sidebar-logo-link {
|
position: relative;
|
isolation: isolate;
|
height: 100%;
|
width: 100%;
|
display: flex;
|
align-items: center;
|
justify-content: center;
|
padding: 0;
|
|
&::before {
|
content: "";
|
position: absolute;
|
inset: 0;
|
background-image: var(--logo-bg-image);
|
background-size: cover;
|
background-position: center;
|
opacity: 0.26;
|
filter: blur(8px) saturate(0.9);
|
transform: scale(1.06);
|
pointer-events: none;
|
z-index: 0;
|
}
|
|
&::after {
|
content: "";
|
position: absolute;
|
inset: 0;
|
background: linear-gradient(180deg, rgba(16, 49, 89, 0.18), rgba(16, 49, 89, 0.08));
|
pointer-events: none;
|
z-index: 0;
|
}
|
}
|
|
.sidebar-logo {
|
width: 100%;
|
height: 100%;
|
max-width: none;
|
max-height: none;
|
padding: 6px 10px;
|
vertical-align: middle;
|
object-fit: contain;
|
object-position: center;
|
filter: none;
|
display: block;
|
position: relative;
|
z-index: 1;
|
}
|
|
.sidebar-title {
|
display: inline-block;
|
margin: 0;
|
color: var(--text-primary);
|
font-weight: 600;
|
line-height: 1.2;
|
font-size: 14px;
|
font-family: "Segoe UI", "PingFang SC", sans-serif;
|
vertical-align: middle;
|
position: relative;
|
z-index: 1;
|
}
|
|
&.collapse {
|
.sidebar-logo-link {
|
display: flex;
|
align-items: center;
|
justify-content: center;
|
padding: 0;
|
|
&::before,
|
&::after {
|
display: none;
|
}
|
}
|
|
.sidebar-logo {
|
width: calc(100% - 8px);
|
height: calc(100% - 8px);
|
max-width: none;
|
max-height: none;
|
padding: 4px;
|
margin: 0 auto;
|
filter: none;
|
object-fit: contain;
|
object-position: center;
|
}
|
|
.sidebar-favicon {
|
width: calc(100% - 8px);
|
height: calc(100% - 8px);
|
max-width: none;
|
max-height: none;
|
}
|
}
|
}
|
</style>
|