From 4057a78368f09b08e08034788dc38c54dd6d5778 Mon Sep 17 00:00:00 2001
From: ljx <15723110242@139.com>
Date: Mon, 15 Sep 2025 18:04:57 +0800
Subject: [PATCH 1/4] =?UTF-8?q?=E4=BF=AE=E6=94=B9=E4=BB=B7=E6=A0=BC?=
 =?UTF-8?q?=E6=A0=BC=E5=BC=8F?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
 .env.development   | 2 +-
 src/utils/ruoyi.ts | 5 +++++
 2 files changed, 6 insertions(+), 1 deletion(-)
diff --git a/.env.development b/.env.development
index b8354ec..15482f5 100644
--- a/.env.development
+++ b/.env.development
@@ -7,7 +7,7 @@ VITE_APP_ENV = 'development'
 # 开发环境
 # VITE_APP_BASE_API = 'http://192.168.110.180:8899'
 # 李陈杰 209
-VITE_APP_BASE_API = 'http://192.168.110.180:8899'
+# VITE_APP_BASE_API = 'http://192.168.110.180:8899'
 # 李陈杰 209
 #  VITE_APP_BASE_API = 'http://192.168.110.209:8899'
 # 曾涛
diff --git a/src/utils/ruoyi.ts b/src/utils/ruoyi.ts
index c92994d..817fbad 100644
--- a/src/utils/ruoyi.ts
+++ b/src/utils/ruoyi.ts
@@ -74,6 +74,11 @@ export const formatPrice = (price, show = true) => {
   const fixedNum = num.toFixed(4);
   const [integer, decimal] = fixedNum.split('.');
 
+  // 检查小数部分是否为0
+  if (decimal === '0000') {
+    return `${integer}.00`;
+  }
+
   // 千分位处理
   const formattedInteger = integer.replace(/\B(?=(\d{3})+(?!\d))/g, ',');
 
From 2ea9d901b5fc9b3cefe8d6c4d85f2a8d90f832e3 Mon Sep 17 00:00:00 2001
From: ljx <15723110242@139.com>
Date: Mon, 15 Sep 2025 19:12:23 +0800
Subject: [PATCH 2/4] =?UTF-8?q?=E4=BF=AE=E6=94=B9=E5=A4=A7=E5=B1=8F?=
 =?UTF-8?q?=E6=A0=B7=E5=BC=8F?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
 .../components/header.vue                     | 29 +++++++++++--------
 src/views/largeScreen/components/header.vue   | 29 +++++++++++--------
 .../ProjectScreen/components/header.vue       | 25 +++++++++-------
 3 files changed, 48 insertions(+), 35 deletions(-)
diff --git a/src/views/enterpriseLarge/digitalizationScreen/components/header.vue b/src/views/enterpriseLarge/digitalizationScreen/components/header.vue
index 56c3916..fe3f97e 100644
--- a/src/views/enterpriseLarge/digitalizationScreen/components/header.vue
+++ b/src/views/enterpriseLarge/digitalizationScreen/components/header.vue
@@ -23,7 +23,7 @@
               v-for="(item, i) in weatherList"
               :key="i"
               class="weather-item"
-              :style="{ transform: `translateY(-${offsetY}px)`, transition: transition }"
+              :style="{ transform: `translateY(-${offsetY}vw)`, transition: transition }"
             >
               ![]() 
               
{{ item.weather }}{{ item.tempMin }}°/{{ item.tempMax }}°
@@ -39,7 +39,7 @@
         
         
           
-          
管理系统
+          
管理系统
          
         
         
@@ -89,7 +89,7 @@ const emit = defineEmits(['changePage']);
 const safetyDay = ref
(0);
 const weatherList = ref([]);
 const timer = ref(0);
-const offsetY = ref(0);
+const offsetY = ref(0);
 const curIndex = ref(0);
 const transition = ref('transform 0.5s ease');
 const pendingPause = ref(false);
@@ -119,7 +119,7 @@ function judgeDayOrNight(sunRise: string, sunSet: string) {
 const setWeatherScroll = () => {
   curIndex.value += 1;
   transition.value = 'transform 0.3s ease';
-  offsetY.value = curIndex.value * 60;
+  offsetY.value = curIndex.value * 2;
 
   if (curIndex.value === weatherList.value.length - 1) {
     setTimeout(() => {
@@ -232,13 +232,13 @@ onUnmounted(() => {
 
 .title > div:first-child {
   /* 第一个子元素的样式 */
-  font-size: 38px;
+  font-size: 2vw;
   letter-spacing: 0.1em;
 }
 
 .title > div:last-child {
   /* 最后一个子元素的样式 */
-  font-size: 14px;
+  font-size: 1vw;
 }
 
 /* 顶部栏容器:Flex 水平布局 + 垂直居中 */
@@ -260,22 +260,25 @@ onUnmounted(() => {
   align-items: center;
 
   .weather-list {
-    height: 60px;
+    height: 2vw;
     overflow: hidden;
 
     .weather-item {
-      height: 60px;
-      line-height: 60px;
+      height: 2vw;
       display: flex;
       align-items: center;
+      justify-content: center;
+      // padding: 10px 0;
+      // box-sizing: border-box;
+      font-size: 0.8vw;
 
       & > div:last-child {
         margin-left: 10px;
       }
 
       img {
-        width: 50px;
-        height: 50px;
+        width: 3vw;
+        height: 3vw;
       }
     }
   }
@@ -305,10 +308,12 @@ onUnmounted(() => {
 
 /* 右侧区域(管理系统):图标 + 文字水平排列 */
 .right-section {
+  width: 5.5vw;
   display: flex;
   align-items: center;
+  justify-content: center;
   font-family: 'AlimamaShuHeiTi', sans-serif;
-  font-size: 20px;
+  font-size: 1vw;
   cursor: pointer;
 }
 
diff --git a/src/views/largeScreen/components/header.vue b/src/views/largeScreen/components/header.vue
index 02848cf..fc782ac 100644
--- a/src/views/largeScreen/components/header.vue
+++ b/src/views/largeScreen/components/header.vue
@@ -23,7 +23,7 @@
               v-for="(item, i) in weatherList"
               :key="i"
               class="weather-item"
-              :style="{ transform: `translateY(-${offsetY}px)`, transition: transition }"
+              :style="{ transform: `translateY(-${offsetY}vw)`, transition: transition }"
             >
               ![]() 
               {{ item.weather }}{{ item.tempMin }}°/{{ item.tempMax }}°
@@ -79,7 +79,7 @@ const userStore = useUserStoreHook();
 const currentProject = computed(() => userStore.selectedProject);
 
 // 天气轮播相关变量
-const weatherList = ref([]);
+const weatherList = ref([]);
 const offsetY = ref(0);
 const curIndex = ref(0);
 const transition = ref('transform 0.5s ease');
@@ -103,7 +103,7 @@ function judgeDayOrNight(sunRise: string, sunSet: string) {
 const setWeatherScroll = () => {
   curIndex.value += 1;
   transition.value = 'transform 0.3s ease';
-  offsetY.value = curIndex.value * 60; // 每个天气项高度60px,需和样式一致
+  offsetY.value = curIndex.value * 2; // 每个天气项高度60px,需和样式一致
 
   // 轮播到最后一项时,无缝衔接回第一项
   if (curIndex.value === weatherList.value.length - 1) {
@@ -160,7 +160,7 @@ const getWeatherData = async () => {
         weatherList.value = res.data;
 
         // 处理每一天的天气(白天/夜晚切换图标和状态)
-        weatherList.value.forEach((item) => {
+        weatherList.value.forEach((item: any) => {
           const isDay = judgeDayOrNight(item.sunRise, item.sunSet);
           item.status = isDay ? item.dayStatus : item.nightStatus;
           item.icon = isDay ? item.dayIcon : item.nightIcon;
@@ -250,13 +250,13 @@ onUnmounted(() => {
 
 .title > div:first-child {
   /* 第一个子元素的样式 */
-  font-size: 38px;
+  font-size: 2vw;
   letter-spacing: 0.1em;
 }
 
 .title > div:last-child {
   /* 最后一个子元素的样式 */
-  font-size: 14px;
+  font-size: 1vw;
 }
 
 /* 顶部栏容器:Flex 水平布局 + 垂直居中 */
@@ -278,22 +278,25 @@ onUnmounted(() => {
   align-items: center;
 
   .weather-list {
-    height: 60px;
+    height: 2vw;
     overflow: hidden;
 
     .weather-item {
-      height: 60px;
-      line-height: 60px;
+      height: 2vw;
       display: flex;
       align-items: center;
+      justify-content: center;
+      // padding: 10px 0;
+      // box-sizing: border-box;
+      font-size: 0.8vw;
 
       & > div:last-child {
         margin-left: 10px;
       }
 
       img {
-        width: 50px;
-        height: 50px;
+        width: 3vw;
+        height: 3vw;
       }
     }
   }
@@ -323,10 +326,12 @@ onUnmounted(() => {
 
 /* 右侧区域(管理系统):图标 + 文字水平排列 */
 .right-section {
+  width: 5.5vw;
   display: flex;
   align-items: center;
+  justify-content: center;
   font-family: 'AlimamaShuHeiTi', sans-serif;
-  font-size: 20px;
+  font-size: 1vw;
   cursor: pointer;
 }
 
diff --git a/src/views/projectLarge/ProjectScreen/components/header.vue b/src/views/projectLarge/ProjectScreen/components/header.vue
index a1e3eb2..b891b0d 100644
--- a/src/views/projectLarge/ProjectScreen/components/header.vue
+++ b/src/views/projectLarge/ProjectScreen/components/header.vue
@@ -23,7 +23,7 @@
               v-for="(item, i) in weatherList"
               :key="i"
               class="weather-item"
-              :style="{ transform: `translateY(-${offsetY}px)`, transition: transition }"
+              :style="{ transform: `translateY(-${offsetY}vw)`, transition: transition }"
             >
               ![]() 
               {{ item.weather }}{{ item.tempMin }}°/{{ item.tempMax }}°
@@ -39,7 +39,7 @@
         
         
           
-          
管理系统
+          
管理系统
          
         
         
@@ -129,7 +129,7 @@ function judgeDayOrNight(sunRise: string, sunSet: string) {
 const setWeatherScroll = () => {
   curIndex.value += 1;
   transition.value = 'transform 0.3s ease';
-  offsetY.value = curIndex.value * 60;
+  offsetY.value = curIndex.value * 2;
 
   if (curIndex.value === weatherList.value.length - 1) {
     setTimeout(() => {
@@ -242,13 +242,13 @@ onUnmounted(() => {
 
 .title > div:first-child {
   /* 第一个子元素的样式 */
-  font-size: 38px;
+  font-size: 2vw;
   letter-spacing: 0.1em;
 }
 
 .title > div:last-child {
   /* 最后一个子元素的样式 */
-  font-size: 26px;
+  font-size: 1.5vw;
 }
 
 /* 顶部栏容器:Flex 水平布局 + 垂直居中 */
@@ -270,22 +270,23 @@ onUnmounted(() => {
   align-items: center;
 
   .weather-list {
-    height: 60px;
+    height: 2vw;
     overflow: hidden;
 
     .weather-item {
-      height: 60px;
-      line-height: 60px;
+      height: 2vw;
       display: flex;
       align-items: center;
+      justify-content: center;
+      font-size: 0.8vw;
 
       & > div:last-child {
         margin-left: 10px;
       }
 
       img {
-        width: 50px;
-        height: 50px;
+        width: 3vw;
+        height: 3vw;
       }
     }
   }
@@ -315,10 +316,12 @@ onUnmounted(() => {
 
 /* 右侧区域(管理系统):图标 + 文字水平排列 */
 .right-section {
+  width: 5.5vw;
   display: flex;
   align-items: center;
+  justify-content: center;
   font-family: 'AlimamaShuHeiTi', sans-serif;
-  font-size: 20px;
+  font-size: 1vw;
   cursor: pointer;
 }
 
From 0b6dcc98e83f91017e0a2fee941861efb4853897 Mon Sep 17 00:00:00 2001
From: taoge1020 
Date: Mon, 15 Sep 2025 20:09:56 +0800
Subject: [PATCH 3/4] =?UTF-8?q?=E4=BC=98=E5=8C=96?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
 .../formalitiesAreConsolidated/index.vue      |  11 +-
 src/views/project/attendance/index.vue        |  28 +-
 .../busSalaryDetails/component/detail.vue     | 102 ++++
 .../busSalaryDetails/component/edit.vue       | 146 +++++
 .../busSalaryDetails/component/model.ts       |  57 ++
 src/views/project/busSalaryDetails/index.vue  | 563 ++++++++++++++++++
 6 files changed, 903 insertions(+), 4 deletions(-)
 create mode 100644 src/views/project/busSalaryDetails/component/detail.vue
 create mode 100644 src/views/project/busSalaryDetails/component/edit.vue
 create mode 100644 src/views/project/busSalaryDetails/component/model.ts
 create mode 100644 src/views/project/busSalaryDetails/index.vue
diff --git a/src/views/formalities/formalitiesAreConsolidated/index.vue b/src/views/formalities/formalitiesAreConsolidated/index.vue
index fad4d4a..2e693de 100644
--- a/src/views/formalities/formalitiesAreConsolidated/index.vue
+++ b/src/views/formalities/formalitiesAreConsolidated/index.vue
@@ -82,11 +82,18 @@
         
       
 
-      
+      
         
         
           
-            {{ scope.row.formalitiesName }}
+            {{ scope.row.formalitiesName }}
           
         
         
diff --git a/src/views/project/attendance/index.vue b/src/views/project/attendance/index.vue
index 9d79f7f..b99239a 100644
--- a/src/views/project/attendance/index.vue
+++ b/src/views/project/attendance/index.vue
@@ -31,6 +31,7 @@
             
               搜索
               重置
+              导出
             
           
         
@@ -234,6 +235,17 @@ const dialog = reactive({
   details: false,
   title: ''
 });
+const now = new Date();
+
+// 获取年份(4位数字)
+const year = now.getFullYear();
+
+// 获取月份(注意:getMonth() 返回 0-11,需要 +1 转换为 1-12)
+const month = now.getMonth() + 1;
+
+// 格式化月份为两位数(不足两位补0),拼接成年月字符串
+const currentYearMonth = `${year}-${month.toString().padStart(2, '0')}`;
+
 const echartsOption = ref({});
 const initFormData: AttendanceForm = {
   id: undefined,
@@ -260,7 +272,7 @@ const data = reactive>({
     pageNum: 1,
     pageSize: 10,
     userName: undefined,
-    clockDate: undefined,
+    clockDate: currentYearMonth,
     clockStatus: undefined,
     commuter: undefined,
     projectId: currentProject.value?.id,
@@ -454,7 +466,19 @@ const init = () => {
     handleQuery();
   });
 };
-
+const onExport = () => {
+  try {
+    console.log(queryParams.value.clockDate);
+    proxy?.download(
+      'project/attendance/exportList',
+      { projectId: currentProject.value?.id, clockDate: queryParams.value.clockDate },
+      `考勤列表_${queryParams.value.clockDate}.xlsx`
+    );
+  } catch (error) {
+    ElMessage.error('导出失败,请重试');
+    console.error('文件导出错误:', error);
+  }
+};
 //监听项目id刷新数据
 const listeningProject = watch(
   () => currentProject.value?.id,
diff --git a/src/views/project/busSalaryDetails/component/detail.vue b/src/views/project/busSalaryDetails/component/detail.vue
new file mode 100644
index 0000000..db5d506
--- /dev/null
+++ b/src/views/project/busSalaryDetails/component/detail.vue
@@ -0,0 +1,102 @@
+
+  
+  
+    
+      
+        员工工资考核记录详情
+      
+      
+        
+        
+          
+            {{ scope.row.dateOfIssue + '-' + scope.row.working_date }}
+          
+        
+        
+        
+      
+    
+  
+
+
+
diff --git a/src/views/project/busSalaryDetails/component/edit.vue b/src/views/project/busSalaryDetails/component/edit.vue
new file mode 100644
index 0000000..0d08744
--- /dev/null
+++ b/src/views/project/busSalaryDetails/component/edit.vue
@@ -0,0 +1,146 @@
+
+  
+    
+    
+      
+        
+          {{ (!formData.id || formData.id == 0 ? '添加' : '修改') + '员工工资考核记录' }}
+        
+      
+       
+      
+        
+      
+    
+  
+    
+      
+        
+          
+            
+          
+          
+            
+              
+            
+          
+          
+            
+              
+            
+          
+          
+            
+          
+          
+            搜索
+            重置
+          
+          批量删除
+          
+          导出员工工资表
+        
+      
+      
+        
+        
+        
+        
+        
+        
+        
+        
+        
+          
+            {{ (Number(scope.row.sumDuration) * Number(scope.row.salary)).toFixed(2) }}
+          
+        
+        
+          
+            {{ scope.row.createdAt }}
+          
+        
+        
+          
+            详情
+            导出工资表
+            删除
+          
+        
+      
+      
+    
+    
+