src/views/reportAnalysis/dataDashboard/index.vue
@@ -1,295 +1,304 @@
<template>
  <div class="scale-container">
    <div class="data-dashboard" :style="{ transform: `scale(${scaleRatio})` }">
    <!-- 全屏按钮 - 移动到左上角 -->
    <button class="fullscreen-btn" @click="toggleFullscreen" :title="isFullscreen ? '退出全屏' : '全屏显示'">
      <svg v-if="!isFullscreen" width="20" height="20" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
        <path d="M8 3v3a2 2 0 0 1-2 2H3m18 0h-3a2 2 0 0 1-2-2V3m0 18v-3a2 2 0 0 1 2-2h3M3 16h3a2 2 0 0 1 2 2v3"/>
      </svg>
      <svg v-else width="20" height="20" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
        <path d="M8 3H5a2 2 0 0 0-2 2v3m18 0V5a2 2 0 0 0-2-2h-3m0 18h3a2 2 0 0 0 2-2v-3M3 16v3a2 2 0 0 0 2 2h3"/>
      </svg>
    </button>
    <!-- 顶部标题栏 -->
    <div class="dashboard-header">
      <div class="factory-name">{{ userStore.currentFactoryName }}</div>
    </div>
    <!-- 主要内容区域 -->
    <div class="dashboard-content">
    <!-- 左侧区域 -->
    <div class="left-panel">
      <!-- 客户信息统计分析 -->
      <LeftTop />
      <!-- 质量统计 -->
      <LeftBottom />
    </div>
    <!-- 中间区域 -->
    <div class="center-panel">
      <CenterTop />
      <CenterBottom />
    </div>
    <!-- 右侧区域 -->
    <div class="right-panel">
      <!-- 应收应付统计 -->
      <RightTop />
      <!-- 回款与开票分析 -->
       <RightBottom />
    </div>
    </div>
    <div class="data-dashboard"
         :style="{ transform: `scale(${scaleRatio})` }">
      <!-- 全屏按钮 - 移动到左上角 -->
      <button class="fullscreen-btn"
              @click="toggleFullscreen"
              :title="isFullscreen ? '退出全屏' : '全屏显示'">
        <svg v-if="!isFullscreen"
             width="20"
             height="20"
             viewBox="0 0 24 24"
             fill="none"
             stroke="currentColor"
             stroke-width="2">
          <path d="M8 3v3a2 2 0 0 1-2 2H3m18 0h-3a2 2 0 0 1-2-2V3m0 18v-3a2 2 0 0 1 2-2h3M3 16h3a2 2 0 0 1 2 2v3" />
        </svg>
        <svg v-else
             width="20"
             height="20"
             viewBox="0 0 24 24"
             fill="none"
             stroke="currentColor"
             stroke-width="2">
          <path d="M8 3H5a2 2 0 0 0-2 2v3m18 0V5a2 2 0 0 0-2-2h-3m0 18h3a2 2 0 0 0 2-2v-3M3 16v3a2 2 0 0 0 2 2h3" />
        </svg>
      </button>
      <!-- 顶部标题栏 -->
      <div class="dashboard-header">
        <div class="factory-name">基础数据</div>
      </div>
      <!-- 主要内容区域 -->
      <div class="dashboard-content">
        <!-- 左侧区域 -->
        <div class="left-panel">
          <LeftTop />
          <LeftBottom />
        </div>
        <!-- 中间区域 -->
        <div class="center-panel">
          <CenterTop />
          <!-- <CenterBottom /> -->
        </div>
        <!-- 右侧区域 -->
        <div class="right-panel">
          <RightTop />
          <RightBottom />
        </div>
      </div>
    </div>
  </div>
</template>
<script setup>
import { ref, onMounted, onBeforeUnmount, nextTick } from 'vue'
import autofit from 'autofit.js'
import LeftTop from './components/basic/left-top.vue'
import LeftBottom from './components/basic/left-bottom.vue'
import CenterTop from './components/basic/center-top.vue'
import CenterBottom from './components/basic/center-bottom.vue'
import RightTop from './components/basic/right-top.vue'
import RightBottom from './components/basic/right-bottom.vue'
import useUserStore from '@/store/modules/user'
  import { ref, onMounted, onBeforeUnmount, nextTick } from "vue";
  import autofit from "autofit.js";
  import LeftTop from "./components/basic/left-top.vue";
  import LeftBottom from "./components/basic/left-bottom.vue";
  import CenterTop from "./components/basic/center-top.vue";
  import CenterBottom from "./components/basic/center-bottom.vue";
  import RightTop from "./components/basic/right-top.vue";
  import RightBottom from "./components/basic/right-bottom.vue";
  import useUserStore from "@/store/modules/user";
// 全屏相关状态
const isFullscreen = ref(false);
  // 全屏相关状态
  const isFullscreen = ref(false);
// 缩放比例
const scaleRatio = ref(1)
// 设计尺寸(基准尺寸)- 根据实际设计稿调整
const designWidth = 1920
const designHeight = 1080
  // 缩放比例
  const scaleRatio = ref(1);
  // 设计尺寸(基准尺寸)- 根据实际设计稿调整
  const designWidth = 1920;
  const designHeight = 1080;
// 用户store
const userStore = useUserStore()
  // 用户store
  const userStore = useUserStore();
// 计算缩放比例
const calculateScale = () => {
  const container = document.querySelector('.scale-container')
  if (!container) return
  // 计算缩放比例
  const calculateScale = () => {
    const container = document.querySelector(".scale-container");
    if (!container) return;
  // 获取容器的实际尺寸
  const rect = container.getBoundingClientRect?.()
  const containerWidth = container.clientWidth || rect?.width || window.innerWidth
  const containerHeight = container.clientHeight || rect?.height || window.innerHeight
    // 获取容器的实际尺寸
    const rect = container.getBoundingClientRect?.();
    const containerWidth =
      container.clientWidth || rect?.width || window.innerWidth;
    const containerHeight =
      container.clientHeight || rect?.height || window.innerHeight;
  // 计算宽高缩放比例,取较小值以保证内容完整显示(等比缩放)
  const scaleX = containerWidth / designWidth
  const scaleY = containerHeight / designHeight
  scaleRatio.value = Math.min(scaleX, scaleY)
}
    // 计算宽高缩放比例,取较小值以保证内容完整显示(等比缩放)
    const scaleX = containerWidth / designWidth;
    const scaleY = containerHeight / designHeight;
    scaleRatio.value = Math.min(scaleX, scaleY);
  };
// 窗口大小变化处理
const handleResize = () => {
  // 延迟执行,确保DOM更新完成
  setTimeout(() => {
    calculateScale()
  }, 100)
}
  // 窗口大小变化处理
  const handleResize = () => {
    // 延迟执行,确保DOM更新完成
    setTimeout(() => {
      calculateScale();
    }, 100);
  };
// 全屏功能实现 - 针对scale-container元素
const toggleFullscreen = () => {
  const element = document.querySelector('.scale-container')
  // 全屏功能实现 - 针对scale-container元素
  const toggleFullscreen = () => {
    const element = document.querySelector(".scale-container");
  if (!element) return
    if (!element) return;
  if (!isFullscreen.value) {
    if (element.requestFullscreen) {
      element.requestFullscreen()
    } else if (element.webkitRequestFullscreen) {
      element.webkitRequestFullscreen()
    } else if (element.msRequestFullscreen) {
      element.msRequestFullscreen()
    if (!isFullscreen.value) {
      if (element.requestFullscreen) {
        element.requestFullscreen();
      } else if (element.webkitRequestFullscreen) {
        element.webkitRequestFullscreen();
      } else if (element.msRequestFullscreen) {
        element.msRequestFullscreen();
      }
    } else {
      if (document.exitFullscreen) {
        document.exitFullscreen();
      } else if (document.webkitExitFullscreen) {
        document.webkitExitFullscreen();
      } else if (document.msExitFullscreen) {
        document.msExitFullscreen();
      }
    }
  } else {
    if (document.exitFullscreen) {
      document.exitFullscreen()
    } else if (document.webkitExitFullscreen) {
      document.webkitExitFullscreen()
    } else if (document.msExitFullscreen) {
      document.msExitFullscreen()
  };
  // 监听全屏变化事件
  const handleFullscreenChange = () => {
    const fullscreenElement =
      document.fullscreenElement ||
      document.webkitFullscreenElement ||
      document.msFullscreenElement;
    isFullscreen.value =
      fullscreenElement &&
      fullscreenElement.classList.contains("scale-container");
    // 全屏状态变化时,延迟重新计算缩放比例(确保DOM更新完成)
    setTimeout(() => {
      calculateScale();
    }, 200);
  };
  // 生命周期钩子
  onMounted(() => {
    // 使用nextTick确保DOM完全渲染后再初始化
    nextTick(() => {
      // 计算初始缩放比例
      calculateScale();
    });
    window.addEventListener("resize", handleResize);
    window.addEventListener("fullscreenchange", handleFullscreenChange);
    window.addEventListener("webkitfullscreenchange", handleFullscreenChange);
    window.addEventListener("MSFullscreenChange", handleFullscreenChange);
  });
  onBeforeUnmount(() => {
    window.removeEventListener("resize", handleResize);
    window.removeEventListener("fullscreenchange", handleFullscreenChange);
    window.removeEventListener("webkitfullscreenchange", handleFullscreenChange);
    window.removeEventListener("MSFullscreenChange", handleFullscreenChange);
    // 移除我们添加的autofit动态调整监听器
    if (window._autofitUpdateHandler) {
      window.removeEventListener("resize", window._autofitUpdateHandler);
      delete window._autofitUpdateHandler;
    }
  }
}
// 监听全屏变化事件
const handleFullscreenChange = () => {
  const fullscreenElement = document.fullscreenElement ||
                           document.webkitFullscreenElement ||
                           document.msFullscreenElement
  isFullscreen.value = fullscreenElement && fullscreenElement.classList.contains('scale-container')
  // 全屏状态变化时,延迟重新计算缩放比例(确保DOM更新完成)
  setTimeout(() => {
    calculateScale()
  }, 200)
}
// 生命周期钩子
onMounted(() => {
  // 使用nextTick确保DOM完全渲染后再初始化
  nextTick(() => {
    // 计算初始缩放比例
    calculateScale()
  })
  window.addEventListener('resize', handleResize)
  window.addEventListener('fullscreenchange', handleFullscreenChange)
  window.addEventListener('webkitfullscreenchange', handleFullscreenChange)
  window.addEventListener('MSFullscreenChange', handleFullscreenChange)
})
onBeforeUnmount(() => {
  window.removeEventListener('resize', handleResize)
  window.removeEventListener('fullscreenchange', handleFullscreenChange)
  window.removeEventListener('webkitfullscreenchange', handleFullscreenChange)
  window.removeEventListener('MSFullscreenChange', handleFullscreenChange)
  // 移除我们添加的autofit动态调整监听器
  if (window._autofitUpdateHandler) {
    window.removeEventListener('resize', window._autofitUpdateHandler)
    delete window._autofitUpdateHandler
  }
  // 关闭autofit
  autofit.off()
})
    // 关闭autofit
    autofit.off();
  });
</script>
<style scoped>
/* 外部缩放容器 - 占据整个视口 */
.scale-container {
position: relative;
width: 100%;
/* 页面在常规布局下(有顶栏)默认减去 84px,避免内容被裁切 */
height: calc(100vh - 84px);
display: flex;
align-items: center;
justify-content: center;
background-color: #000;
overflow: hidden;
}
  /* 外部缩放容器 - 占据整个视口 */
  .scale-container {
    position: relative;
    width: 100%;
    /* 页面在常规布局下(有顶栏)默认减去 84px,避免内容被裁切 */
    height: calc(100vh - 84px);
    display: flex;
    align-items: center;
    justify-content: center;
    background-color: #000;
    overflow: hidden;
  }
/* 内部内容区域 - 固定设计尺寸 */
.data-dashboard {
position: relative;
width: 1920px;
height: 1080px;
background-image: url("@/assets/BI/backImage@2x.png");
background-size: cover;
background-position: center;
background-repeat: no-repeat;
transform-origin: center center;
}
  /* 内部内容区域 - 固定设计尺寸 */
  .data-dashboard {
    position: relative;
    width: 1920px;
    height: 1080px;
    background-image: url("@/assets/BI/backImage@2x.png");
    background-size: cover;
    background-position: center;
    background-repeat: no-repeat;
    transform-origin: center center;
  }
/* 全屏状态的样式 - 作用于scale-container */
.scale-container:fullscreen {
width: 100vw;
height: 100vh;
margin: 0;
padding: 0;
background-color: #000;
z-index: 9999;
}
  /* 全屏状态的样式 - 作用于scale-container */
  .scale-container:fullscreen {
    width: 100vw;
    height: 100vh;
    margin: 0;
    padding: 0;
    background-color: #000;
    z-index: 9999;
  }
/* Webkit浏览器前缀 */
.scale-container:-webkit-full-screen {
width: 100vw;
height: 100vh;
margin: 0;
padding: 0;
background-color: #000;
z-index: 9999;
}
  /* Webkit浏览器前缀 */
  .scale-container:-webkit-full-screen {
    width: 100vw;
    height: 100vh;
    margin: 0;
    padding: 0;
    background-color: #000;
    z-index: 9999;
  }
/* MS浏览器前缀 */
.scale-container:-ms-fullscreen {
width: 100vw;
height: 100vh;
margin: 0;
padding: 0;
background-color: #000;
z-index: 9999;
}
  /* MS浏览器前缀 */
  .scale-container:-ms-fullscreen {
    width: 100vw;
    height: 100vh;
    margin: 0;
    padding: 0;
    background-color: #000;
    z-index: 9999;
  }
  .dashboard-header {
    position: relative;
    z-index: 1;
    height: 86px;
    background-image: url("@/assets/BI/biaoti.png");
    background-size: cover;
    background-repeat: no-repeat;
    display: flex;
    align-items: center;
    justify-content: center;
  }
.dashboard-header {
position: relative;
z-index: 1;
height: 86px;
background-image: url("@/assets/BI/biaoti.png");
background-size: cover;
background-repeat: no-repeat;
display: flex;
align-items: center;
justify-content: center;
}
  .factory-name {
    font-weight: 600;
    font-size: 52px;
    color: #ffffff;
    top: 16px;
    position: absolute;
  }
.factory-name {
font-weight: 600;
font-size: 52px;
color: #FFFFFF;
top: 16px;
position: absolute;
}
  .fullscreen-btn {
    position: absolute;
    top: 10px;
    left: 20px;
    width: 40px;
    height: 40px;
    background: rgba(0, 20, 60, 0.8);
    border: 1px solid rgba(0, 212, 255, 0.3);
    border-radius: 6px;
    color: #00d4ff;
    cursor: pointer;
    display: flex;
    align-items: center;
    justify-content: center;
    transition: all 0.3s;
    z-index: 10000;
  }
.fullscreen-btn {
position: absolute;
top: 10px;
left: 20px;
width: 40px;
height: 40px;
background: rgba(0, 20, 60, 0.8);
border: 1px solid rgba(0, 212, 255, 0.3);
border-radius: 6px;
color: #00d4ff;
cursor: pointer;
display: flex;
align-items: center;
justify-content: center;
transition: all 0.3s;
z-index: 10000;
}
  .fullscreen-btn:hover {
    background: rgba(0, 30, 90, 0.9);
    border-color: rgba(0, 212, 255, 0.5);
  }
.fullscreen-btn:hover {
background: rgba(0, 30, 90, 0.9);
border-color: rgba(0, 212, 255, 0.5);
}
  .dashboard-content {
    position: relative;
    z-index: 1;
    display: flex;
    gap: 30px;
    padding: 0 30px;
    height: calc(100% - 86px);
    overflow: hidden;
  }
.dashboard-content {
position: relative;
z-index: 1;
display: flex;
gap: 30px;
padding: 0 30px;
height: calc(100% - 86px);
overflow: hidden;
}
  /* 确保各面板能够正确显示 */
  .left-panel,
  .center-panel,
  .right-panel {
    overflow: hidden;
  }
/* 确保各面板能够正确显示 */
.left-panel, .center-panel, .right-panel {
overflow: hidden;
}
  .left-panel,
  .right-panel {
    flex: 1;
    display: flex;
    flex-direction: column;
    gap: 24px;
    width: 520px;
  }
.left-panel,
.right-panel {
flex: 1;
display: flex;
flex-direction: column;
gap: 24px;
width: 520px;
}
.center-panel {
flex: 1.5;
display: flex;
flex-direction: column;
gap: 20px;
}
  .center-panel {
    flex: 1.5;
    display: flex;
    flex-direction: column;
    gap: 20px;
  }
</style>