Browse Source

日报统计完成

wangtao 1 month ago
parent
commit
0af2424794

+ 10 - 0
src/api/business/project.js

@@ -140,6 +140,16 @@ export function projectApi() {
                 params
             })
         },
+        //获取日报统计时间
+        timeQuery(params) {
+            return request({
+                url: '/service-iot/pmTimeConf/timeQuery',
+                method: 'GET',
+                params
+            })
+        },
+
+        
         
     }
 }

+ 7 - 0
src/api/system/user.js

@@ -100,3 +100,10 @@ export function listDept(query) {
       params: query
   })
 }
+// 查询用户所属企业
+export function getTenantByUser(params) {
+  return request({
+      url: '/system/sysUserTenant/getTenantByUser/' + params,
+      method: 'get',
+  })
+}

+ 1 - 2
src/config.js

@@ -9,8 +9,7 @@ export default {
   //#endif
 
   //#ifdef H5
-  baseUrl: "https://gateWay.usky.cn/prod-api",
-  // baseUrl: import.meta.env.MODE === "production" ? `https://${window.location.host}/prod-api` : `http://192.168.10.165:801/dev-api`,
+  baseUrl: import.meta.env.MODE === "production" ? `https://${window.location.host}/prod-api` : `http://192.168.10.165:801/dev-api`,
   //#endif
 
   websiteUrl: "https://qhome.usky.cn",

+ 1 - 1
src/pages/business/common/projectMange/list/index.vue

@@ -108,7 +108,7 @@
     </template>
   </oa-scroll>
 
-  <oa-tabbar :tabbarValue="2" :tabbarList="proxy.$constData.projectTabbar" :isSwitchTab="false"></oa-tabbar>
+  <oa-tabbar :tabbarValue="3" :tabbarList="proxy.$constData.projectTabbar" :isSwitchTab="false"></oa-tabbar>
 
   <u-popup :show="popup.show" mode="bottom" bgColor="#fff" :round="10" @close="popup.show = false">
     <view

+ 17 - 25
src/pages/business/common/projectMange/record/details.vue

@@ -1,5 +1,5 @@
 <template>
-  <u-navbar v-if="dataList[0]?.createBy" :titleStyle="{ color: '#000' }" :autoBack="true" :title="dataList[0]?.createBy + '的日报'" :placeholder="true" :safeAreaInsetTop="true" bgColor="#fff">
+  <u-navbar  :titleStyle="{ color: '#000' }" :autoBack="true" title="日报详情" :placeholder="true" :safeAreaInsetTop="true" bgColor="#fff">
     <template #left>
       <view class="u-navbar__content__left__item">
         <u-icon name="arrow-left" size="20" color="#000"></u-icon>
@@ -14,10 +14,10 @@
     :isSticky="true"
     :customStyle="{
       //#ifdef APP-PLUS || MP-WEIXIN
-      height: `calc(100vh - (138px + ${proxy.$settingStore.StatusBarHeight}))`,
+      height: `calc(100vh - 44px)`,
       //#endif
       //#ifdef H5
-      height: `calc(100vh - (138px))`,
+      height: `calc(100vh - 44px)`,
       //#endif
     }"
     :refresherLoad="true"
@@ -33,9 +33,9 @@
     <template #default>
       <view class="content-area radius bg-white" v-for="(item, index) in dataList" :key="index" style="margin-top: 6px">
         <view class="content-area-header flex mb10">
-          <img :src="item.avatar" class="content-area-header-avatarImg mr10" v-if="item.avatar" />
+          <img :src="item?.avatar" class="content-area-header-avatarImg mr10" v-if="item?.avatar" />
           <u-avatar
-            v-if="!item.avatar"
+            v-if="!item?.avatar"
             class="content-area-header-avatar mr10"
             :text="item.createBy.length > 2 ? item.createBy.slice(1, 3) : item.createBy"
             shape="square"
@@ -92,7 +92,7 @@
             style="width: 100%"
           ></u-tabs>
           <view class="content-area-header mt20 mb10 text-center" style="display: inline-block" v-for="(item, index) in reportDetailData.treeSelectNodes" :key="index">
-            <img v-if="item.avatar" class="content-area-header-avatarImg mlr5" :src="item.avatar" style="display: block; width: 40px; height: 40px" />
+            <img v-if="item?.avatar" class="content-area-header-avatarImg mlr5" :src="item.avatar" style="display: block; width: 40px; height: 40px" />
             <u-avatar
               v-if="!item.avatar"
               class="content-area-header-avatar mlr5"
@@ -217,18 +217,16 @@ function toProjectMange(id) {
  * @scrollView加载数据
  */
  function load() {
+  console.log(1111)
   projectApi().ReportRecord(
     {
       upOrDown:0,
       slideSum:1,
-      submitDate:state.submitDate,
+      submitDate:dataList.value[dataList.value.length-1].submitDate,
       projectAscription:2
     }
   ).then((requset) => {
-    dataList.value.push(requset.data.records[0]);
-    if(dataList.value.length>2){
-      delete dataList.value[0]
-    }
+    dataList.value.push(requset.data.records[0])
   })
 }
 /**
@@ -239,20 +237,14 @@ function toProjectMange(id) {
     {
       upOrDown:1,
       slideSum:1,
-      submitDate:state.submitDate,
+      submitDate:dataList.value[0].submitDate,
       projectAscription:2
     }
   ).then((requset) => {
-    dataList.value.unshift(requset.data.records[0]);
-    state.submitDate = requset.data.records[0].submitDate;
-    if(dataList.value.length>2){
-      delete dataList.value[2]
+    if(requset.data.records.length>0){
+      dataList.value.unshift(requset.data.records[0]);
     }
   })
-  // console.log(dataList.value)
-  
-  // state.pageSize = 20;
-  // init();
 }
 onReady(() => {});
 
@@ -309,9 +301,9 @@ onUnload(() => {
     }
   }
 }
-.uni-scroll-view-content .content-area:nth-child(3){
-  height:96px;
-  overflow: hidden;
-  background-image: linear-gradient(to top, #f3f2f2 30%, #fff 100%);
-}
+// .uni-scroll-view-content .content-area:nth-child(3){
+//   height:96px;
+//   overflow: hidden;
+//   background-image: linear-gradient(to top, #f3f2f2 30%, #fff 100%);
+// }
 </style>

+ 188 - 148
src/pages/business/common/projectMange/statistics/dailyReport.vue

@@ -1,10 +1,8 @@
 <template>
     <u-sticky class="shadow-default" bgColor="#fff" style="top: 0">
-      <u-navbar :titleStyle="{ color: '#000' }" :autoBack="true" title="日报" :placeholder="true" :safeAreaInsetTop="true" bgColor="#fff">
+      <u-navbar :titleStyle="{ color: '#000' }" :autoBack="false" title="日报" :placeholder="true" :safeAreaInsetTop="true" bgColor="#fff">
         <template #left>
-          <view class="u-navbar__content__left__item">
-            <u-icon name="arrow-left" size="20" color="#000"></u-icon>
-          </view>
+            <u-icon name="arrow-left" size="20" color="#000" @click="returnTo('business/common/projectMange/statistics/index')"></u-icon>
         </template>
       </u-navbar>
     </u-sticky>
@@ -24,103 +22,62 @@
           </view>
         </view>
       </view>
-    </view>
-    <oa-scroll
-    customClass="record-container scroll-height"
-    :pageSize="query.pageSize"
-    :total="total"
-    :isSticky="true"
-    :customStyle="{
-      //#ifdef APP-PLUS || MP-WEIXIN
-      height: `calc(100vh - (138px + ${proxy.$settingStore.StatusBarHeight}))`,
-      //#endif
-      //#ifdef H5
-      height: `calc(100vh - (138px))`,
-      //#endif
-    }"
-    :refresherLoad="true"
-    :refresherEnabled="true"
-    :refresherDefaultStyle="'none'"
-    :refresherThreshold="44"
-    :lowerThreshold="44"
-    :refresherBackground="'#f5f6f7'"
-    @load="load"
-    @refresh="refresh"
-    :data-theme="'theme-' + proxy.$settingStore.themeColor.name"
-  >
-    <template #default>
-      <!-- <u-loading-page :loading="state.loading" fontSize="16" style="z-index: 99"></u-loading-page> -->
-      <!-- start -->
-      <view class="content-area" v-for="(el, index) in reportList" :key="index">
-        <!-- <view class="flex mb10" @click="goContentDetails(el)">
-            <img :src="el.avatar" class="content-area-center-avatarImg mr10" v-if="el.avatar" />
+      <view class="report-list">
+        <view class="report-cell" v-for="(el,index) in reportList" :key="index">
+          <view class="flex report" @click="toPage(el.id,el.lockStatus)">
+            <image :src="el.avatar" class="content-area-center-avatarImg mr10" v-if="el.avatar" />
             <u-avatar
-              v-if="!el.avatar"
+              v-if="!el.avatar && query.queryType != 2 && el.createBy"
               class="content-area-center-avatar mr10"
               :text="el.createBy.length > 2 ? el.createBy.slice(1, 3) : el.createBy"
               shape="square"
-              size="35"
-              fontSize="10"
+              fontSize="12"
               color="#ffffff"
               :bgColor="proxy.$settingStore.themeColor.color"
             ></u-avatar>
-            <view>
-              <view class="content-area-center-title font14 mb5">{{ el.createBy }}的日报</view>
-              <view class="content-area-center-time font12">{{ proxy.$time.jktTimes(el.submitDate.replace("T", " ")) }}</view>
-            </view>
-          </view> -->
-        <!-- <view class="content-area-center bg-white" v-else style="margin: 0 10px 10px; border-radius: 10px">
-          <view class="content-area-top menu-item" style="float: right; padding: 10px 0px">
-            <view class="content-area-top-time"> </view>
-            <u-icon class="content-area-top-icon" name="more-dot-fill" size="20" color="#000" @click="moreClick(el)"></u-icon>
-          </view>
-          <view class="flex mb10" @click="goContentDetails(el)">
-            <img :src="el.avatar" class="content-area-center-avatarImg mr10" v-if="el.avatar" />
             <u-avatar
-              v-if="!el.avatar"
+              v-if="!el.avatar && query.queryType == 2 && el.userName"
               class="content-area-center-avatar mr10"
-              :text="el.createBy.length > 2 ? el.createBy.slice(1, 3) : el.createBy"
+              :text="el.nickName.length > 2 ? el.nickName.slice(1, 3) : el.nickName"
               shape="square"
-              size="35"
-              fontSize="10"
+              fontSize="12"
               color="#ffffff"
               :bgColor="proxy.$settingStore.themeColor.color"
             ></u-avatar>
-            <view>
-              <view class="content-area-center-title font14 mb5">{{ el.createBy }}的日报</view>
-              <view class="content-area-center-time font12">{{ proxy.$time.jktTimes(el.submitDate.replace("T", " ")) }}</view>
+            <view class="left">
+              <view class="content-area-center-title font14 mb5">{{ query.queryType == 2 ?  el.nickName : el.createBy }}</view>
+              <view class="content-area-center-time font12">{{ el.deptName }}</view>
+            </view>
+            <view class="right" v-if="query.queryType != 2">
+              <view class="content-area-center-time font12">{{ el.submitDateOutput }}</view>
+              <image src="@/static/images/projectMange/lock.png" alt="" style="width: 10px;height: 12.5px;margin-top:10px;" v-if="el.lockStatus" />
+              <view class="content-area-center-time font12" v-if="!el.lockStatus" :style="{color: el.readStatus ? '#999' : '#2A98FF',marginTop: '10px'}">{{ el.readStatus ? '已读' : '未读' }}</view>
             </view>
           </view>
-          <view class="mb5" @click="goContentDetails(el)">
-            <u-text :text="el.contentText.length >= 100 ? el.contentText.slice(0, 100) + '···' : el.contentText" color="#666666" size="14"></u-text>
-          </view>
-          <view class="flex" v-if="el.createBy != useStore.nickName" @click="goContentDetails(el)">
-            <u-tag class="mr10" type="info" text="已读" size="mini" plain v-if="el.readFlag === 1" style="margin: 0 auto"></u-tag>
-            <u-tag class="mr10" type="error" text="未读" size="mini" plain v-if="el.readFlag === 0" style="margin: 0 auto"></u-tag>
-            <u-text text="全文" :color="proxy.$settingStore.themeColor.color" size="14"></u-text>
-          </view>
-        </view> -->
+        </view>
+        <view class="noData" v-if="!total">
+          <image src="@/static/images/data.png" alt="" />
+        </view>
       </view>
-      <!-- end -->
-    </template>
-  </oa-scroll>
+    </view>
 </template>
   
   <script setup>
   /*----------------------------------依赖引入-----------------------------------*/
   import { onLoad, onShow, onReady } from "@dcloudio/uni-app";
   import { ref, getCurrentInstance, reactive } from "vue";
-  import { useRouter, useRoute } from "vue-router";
   /*----------------------------------接口引入-----------------------------------*/
   import { projectApi } from "@/api/business/project.js";
+  import { listDept } from "@/api/system/user";
   // import dayjs from 'dayjs/esm/index'
   /*----------------------------------组件引入-----------------------------------*/
   /*----------------------------------store引入-----------------------------------*/
+  import { useStores } from "@/store/modules/index";
   import dayjs from "dayjs";
-  const route = useRoute();
   /*----------------------------------公共方法引入-----------------------------------*/
   /*----------------------------------公共变量-----------------------------------*/
   const { proxy } = getCurrentInstance();
+  const useStore = useStores();
   const day = ref("")
   const statusNext = ref(false)//下一天按钮状态
   const currentDate = ref("")//当前日期
@@ -136,10 +93,10 @@
   ])
   const query = ref({
     queryType:0,
-    submitDate:dayjs().format("YYYY-MM-DD"),
+    submitDate:null,
     reportId:null,
     pageNum:1,
-    pageSize:10,
+    pageSize:1000,
   })
   /*----------------------------------变量声明-----------------------------------*/
   const statistics = ref({
@@ -148,26 +105,44 @@
     notSubmitted:0,  //未交
   })
   const loading = ref(false)
-  const reportList =reactive([])
+  const reportList = ref([])
   const pageSize = ref(20)
   const current = ref(1)
   const total = ref(0)
+  var pages = getCurrentPages(); //获取当前页面栈的实例;
   /**
    * @页面初始化
    */
-  function init(id) {
+  function init(id,statisticalDate) {
     if(id){
       query.value.queryType = id
     }
-    var time = dayjs().format("YYYY-MM-DD").split("-")
-    currentDate.value = `${time[0]}${time[1]}${time[2]}`
-    day.value = `${time[0]}年${time[1]}月${time[2]}日`
+    var time;
+    if(statisticalDate){
+      time = statisticalDate.split("-")
+      day.value = `${time[0]}年${time[1]>=10 ? time[1] : time[1].slice(1)}月${time[2]>=10 ? time[2] : time[2].slice(1)}日`
+      query.value.submitDate = statisticalDate
+    }else{
+      time = dayjs().format("YYYY-MM-DD").split("-")
+      day.value = `${time[0]}年${time[1]}月${time[2]}日`
+      query.value.submitDate = dayjs().format("YYYY-MM-DD")
+    }
+    currentDate.value = dayjs().format("YYYYMMDD")
     date.value = {
       year:time[0],
       month:time[1],
       day:time[2]
     }
-    getStatistics()
+    if(`${time[0]}${time[1]}${time[2]}` < currentDate.value){
+      statusNext.value = true
+    }else{
+      statusNext.value = false
+    }
+    if(statisticalDate){
+      getStatistics(statisticalDate)
+    }else{
+      getStatistics()
+    }
     getPageList()
   }
   //获取统计数据
@@ -178,17 +153,36 @@
   }
   //获取分页数据
   function getPageList() {
+    reportList.value = []
     projectApi().pmTimeConfPage(query.value).then((res) => {
-      reportList.value = res.data.records[0];
-      console.log(reportList.value)
+      if(res.data.records){
+        reportList.value = res.data.records[0];
+        total.value = res.data.total;
+        var deptItem = []
+        listDept().then((res)=>{//部门名称
+          deptItem = res.data
+          for(let i=0;i<reportList.value.length;i++){
+            reportList.value[i].deptName = deptItem.find(item => item.deptId == reportList.value[i].deptId).deptName
+            if(query.value.queryType != 2){ //按时提交/迟交
+              var monthDayWrap = reportList.value[i].submitDate.split(" ")[0]
+              var time = reportList.value[i].submitDate.split(" ")[1].slice(0,5)
+              var month = monthDayWrap.split("-")[1] >= 10 ? monthDayWrap.split("-")[1] : monthDayWrap.split("-")[1].slice(1)
+              var day = monthDayWrap.split("-")[2] >= 10 ? monthDayWrap.split("-")[2] : monthDayWrap.split("-")[2].slice(1)
+              reportList.value[i].submitDateOutput = `${month}月${day}日 ${time}` //提交时间
+              reportList.value[i].lockStatus = reportList.value[i].ccTo.indexOf(useStore.userId) > -1 ? false : true //锁定状态
+              reportList.value[i].readStatus = reportList.value[i].pmReportReaders.readAlready.map((item) => item).length > 0 ? true : false //阅读状态
+              if(reportList.value[i].createBy == useStore.userName){//本人
+                reportList.value[i].lockStatus = false
+                reportList.value[i].readStatus = true
+              }
+            }
+          }
+        })
+      }
     });
   }
   /** 日期选择*/
   function daySwitch(id){
-    //获取前一天/后一天日期
-      if(id == 1 && statusNext.value == false){
-        return
-      }
       const targetDate = dayjs(`${date.value.year}-${date.value.month}-${date.value.day}`);
       var previousDay;
       if(id == 0){
@@ -197,63 +191,70 @@
       if(id == 1){
         previousDay = targetDate.subtract(-1, 'day');
       }
-      var time = previousDay.format("YYYY-MM-DD").split("-")
-      day.value = `${time[0]}年${time[1]}月${time[2]}日`
-      date.value = {
-        year:time[0],
-        month:time[1],
-        day:time[2]
+      var time = previousDay.format("YYYY-MM-DD")
+      var time2 = previousDay.format("YYYYMMDD")
+      if(id == 0){
+        uni.redirectTo({
+          url:`/pages/business/common/projectMange/statistics/dailyReport?id=${query.value.queryType}&statisticalDate=${time}`
+        })
       }
-      query.value.submitDate = `${time[0]}-${time[1]}-${time[2]}`
-    //获取下一天按钮状态
-      var changeDate = `${date.value.year}${date.value.month}${date.value.day}`
-      if(changeDate >= currentDate.value){
-        statusNext.value = false
-      }else{
-        statusNext.value = true
+      if(id == 1 && time2 <= currentDate.value){
+        uni.redirectTo({
+          url:`/pages/business/common/projectMange/statistics/dailyReport?id=${query.value.queryType}&statisticalDate=${time}`
+        })
       }
+  }
+  /**类型选择 */
+  function tabsClick(id){
+    query.value.queryType = id
     getStatistics(query.value.submitDate)
     getPageList()
   }
-  /**
-   * @scrollView加载数据
-   */
-  function load() {
-    query.value.current ++;
-    init();
+  /** 跳转到详情页 */
+  function toPage(id,permission){
+    if(query.value.queryType != 2){
+      if(!permission){
+        proxy.$tab.navigateTo(`/pages/business/common/projectMange/record/details?reportId=${id}`);
+      }else{
+        uni.showToast({
+          title: '你不是接收人不能查看内容',
+          icon: "none",
+          duration: 2000,
+        });
+      }
+    }
   }
   /**
-   * @scrollView刷新数据
-   */
-  function refresh() {
-    query.value.current = 1;
-    init();
+ * 返回上级页面
+ * @param defaultPage 默认页面
+ */
+  function returnTo(defaultPage) {
+    if(getCurrentPages().length > 1){
+      uni.navigateBack()
+    }else{
+      uni.switchTab({
+        url: `/pages/${defaultPage}`
+      })
+    }
   }
-
-
-  onLoad(() => {
-  });
-  
-  onShow(() => {
-    if(route?.query?.id){
-      init(route.query.id)
+  onLoad((options) => {
+    if(options?.id && options?.statisticalDate){
+      init(options.id,options.statisticalDate)
+    }else if(options?.id){
+      init(options.id)
+    }else if(options?.statisticalDate){
+      init(null,options.statisticalDate)
     }else{
       init()
     }
+  });
+  
+  onShow(() => {
     //调用系统主题颜色
     proxy.$settingStore.systemThemeColor([1]);
   });
   </script>
   <style lang="scss" scoped>
-  :deep(.u-modal__content) {
-    font-size: 14px;
-    justify-content: left;
-  }
-  :deep(.list-container .content-area-top-name) {
-    font-size: 16px !important;
-  }
-  </style>
-  <style lang="scss" scoped>
   .timeSelect{
     width:100%;
     height:35px;
@@ -290,29 +291,6 @@
     width:100%;
     height:calc(100% - 20px);
     margin:10px 0;
-    .statisticsSearchBox {
-      width: 100%;
-      vertical-align: middle;
-      position: relative;
-      display: flex;
-      .uni-input-input {
-        font-size: 12px !important;
-      }
-      :deep(.u-input__content__prefix-icon) {
-        position: absolute;
-        right: 8px;
-        .uicon-search {
-          font-size: 16px;
-        }
-      }
-    }
-    .u-input {
-      display: inline-block;
-      background: #fff;
-      height: 34px;
-      line-height: 34px;
-      border-radius: 10px
-    }
     .menu-list{
       width:100%;
       margin:0px 0 0 0px;
@@ -367,5 +345,67 @@
       }
     }
   }
+  .report-list{
+    height:calc(100vh - 160px);
+    padding-bottom:50px;
+    overflow: scroll;
+    .report{
+      padding:15px 10px;
+      background: #ffffff;
+      margin-top:4px;
+      .left,.right{
+        width:calc(50% - 25px);
+      }
+      .right{
+        text-align: right;
+      }
+    }
+    .noData{
+      image{
+        width:50%;
+        margin:20px 25%;
+      }
+    }
+  }
+
+  .content-area {
+  &-time {
+    padding: 10px;
+    text-align: left;
+    color: #000000;
+    font-weight: 600;
+  }
+
+  &-center {
+    margin: 0;
+    padding: 15px;
+    overflow: hidden;
+    border-bottom: 1px solid #eaeef1;
+    &:last-child {
+      border-bottom: 0px solid #eaeef1;
+    }
+    &-avatar {
+      margin: auto 0;
+      width:40px !important;
+      height:40px !important;
+    }
+    &-avatarImg {
+      width:40px !important;
+      height:40px !important;
+      border-radius: 4px;
+      >img{
+        width:100% !important;
+      }
+    }
+    &-title {
+      margin: 0 0 15px 0;
+      font-weight: 600;
+      color: #000000;
+    }
+  }
+}
+.list-cell::after{
+  border:none;
+}
   </style>
   

+ 27 - 11
src/pages/business/common/projectMange/statistics/index.vue

@@ -1,22 +1,17 @@
 <template>
   <u-sticky class="shadow-default" bgColor="#fff" style="top: 0">
-    <u-navbar :titleStyle="{ color: '#000' }" :autoBack="true" title="报告统计" :placeholder="true" :safeAreaInsetTop="true" bgColor="#fff">
+    <u-navbar :titleStyle="{ color: '#000' }" :autoBack="false" title="报告统计" :placeholder="true" :safeAreaInsetTop="true" bgColor="#fff">
       <template #left>
-        <view class="u-navbar__content__left__item">
-          <u-icon name="arrow-left" size="20" color="#000"></u-icon>
-        </view>
+          <u-icon name="arrow-left" size="20" color="#000" @click="returnTo('index')"></u-icon>
       </template>
     </u-navbar>
   </u-sticky>
   <view class="container">
-    <!-- <view class="statisticsSearchBox">
-      <u-input v-model="projectName" placeholder="请输入时间" clearable prefixIcon="search" size="small" />
-    </view> -->
     <view style="width: calc(100% - 51px); display: flex; padding-right: 10px">
-          <view>统计时间:18:00 ~ 22:00</view>
+          <view v-if="timeQuery">统计时间:{{ timeQuery.startTime }} ~ {{ timeQuery.onTime }}</view>
         </view>
     <view class="menu-list">
-      <view class="title">{{ statistics?.statisticalDate == currentDate ? '今日' : statistics?.statisticalDate }}报告</view>
+      <view class="title">{{ statistics?.statisticalDate }}</view>
       <view class="list-cell">
         <view class="list"  @click="toPage(item.value)" v-for="(item,index) in activeList" :key="index">
             <text>{{ index == 0 ? statistics.submitOnTime :index == 1 ? statistics.submitLate : index == 2 ? statistics.notSubmitted : '' }}</text> 
@@ -26,6 +21,7 @@
       </view>
     </view>
   </view>
+  <oa-tabbar :tabbarValue="2" :tabbarList="proxy.$constData.projectTabbar" :isSwitchTab="false"></oa-tabbar>
 </template>
 
 <script setup>
@@ -52,11 +48,16 @@ const activeList = reactive([
   {name:"迟交",value:1},
   {name:"未提交",value:2},
 ])
+const timeQuery = ref(null);
 const currentDate = dayjs().format('YYYY-MM-DD');
 /**
  * @页面初始化
  */
 function init() {
+  projectApi().timeQuery().then((res) => {
+    timeQuery.value = res.data;
+  });
+  
   projectApi().pmTimeConfCount().then((res) => {
     statistics.value = res.data;
   });
@@ -66,9 +67,21 @@ function init() {
  * @param id 项目id
  */
 function toPage(id) {
-  proxy.$tab.navigateTo(`/pages/business/common/projectMange/statistics/dailyReport?id=${id}&statisticalDate=2025-02-20`);
+  proxy.$tab.navigateTo(`/pages/business/common/projectMange/statistics/dailyReport?id=${id}&statisticalDate=${statistics.value.statisticalDate}`);
+}
+/**
+ * 返回上级页面
+ * @param defaultPage 默认页面
+ */
+  function returnTo(defaultPage) {
+    if(getCurrentPages().length > 1){
+      uni.navigateBack()
+    }else{
+      uni.switchTab({
+        url: `/pages/${defaultPage}`
+      })
+    }
 }
-
 onReady(() => {});
 
 onShow(() => {
@@ -164,6 +177,9 @@ onShow(() => {
         }
       }
     }
+    .list-cell::after{
+      border:none;
+    }
   }
 }
 </style>

+ 58 - 1
src/pages/index.vue

@@ -27,6 +27,7 @@
     :refresherThreshold="44"
     :refresherBackground="'#f5f6f7'"
     @refresh="refresh"
+    @touchstart="touchStart" @touchend="touchEnd"
   >
     <template #default>
       <view class="home-container">
@@ -76,7 +77,20 @@
       </view>
     </template>
   </oa-scroll>
+  <uni-drawer ref="showLeft" mode="left" :width="320" @change="change($event,'showLeft')">
+    <!-- <view class="close">
+      <button @click="closeDrawer('showLeft')"><text class="word-btn-white">X</text></button>
+    </view> -->
+    <text>切换企业</text>
+    <view class="unit">
+      <view class="list">
+        <u-radio-group v-model="tenantIdChange" @change="changeTenantId">
+          <u-radio :activeColor="proxy.$settingStore.themeColor.color" :name="item.id" :label="item.tenantName" v-for="(item, index) in tenantIdList" :key="index"></u-radio>
+        </u-radio-group>
+      </view>
+    </view>
 
+  </uni-drawer>
   <oa-tabbar :tabbarValue="0" :tabbarList="proxy.$constData.homeTabbar"></oa-tabbar>
 </template>
 
@@ -93,10 +107,16 @@ import { useStores } from "@/store/modules/index";
 import * as jwx from "@/utils/jssdk.js"; //引入js sdk的封装
 import { getToken, setToken, removeToken } from "@/utils/auth";
 import { storage, storageSystem } from "@/utils/storage"; // 公共方法引用
+import { getTenantByUser } from "@/api/system/user.js";
+import { decrypt } from "@/plugins/jsencrypt";
 /*----------------------------------公共变量-----------------------------------*/
 const { proxy } = getCurrentInstance();
 const useStore = useStores();
 /*----------------------------------变量声明-----------------------------------*/
+const touchStartX = ref(0); // 触摸开始X轴坐标
+const showLeft = ref(false); // 左侧菜单显示状态
+const tenantIdChange = ref(""); // 切换租户ID
+const tenantIdList = ref([]); // 租户ID列表
 const state = reactive({
   dialogFlag: false,
 
@@ -108,13 +128,50 @@ const state = reactive({
   cuIconList: [],
   recentlyUsed: [],
 });
-
 const { dialogFlag } = toRefs(state);
 
+function touchStart(e){
+  touchStartX.value = e.touches[0].clientX;
+}
+// 触摸结束时触发此函数
+function touchEnd(e) {
+    // 计算触摸结束时与开始时的坐标差值
+    let deltaX = e.changedTouches[0].clientX - touchStartX.value;
+    if(deltaX > 100){
+      showDrawer()
+    }
+}
+// 单位打开窗口
+function showDrawer() {
+  // showLeft.value.open()
+}
+// 单位关闭窗口
+function 	closeDrawer(e) {
+  showLeft.value.close()
+}
+//获取企业列表
+function getTenantList(id){
+  getTenantByUser(id).then((res) => {
+    tenantIdList.value = res.data
+  })
+}
+function changeTenantId(){
+  // console.log(tenantIdChange.value,12)
+  if(useStore.userName && useStore.userId){
+
+  }
+}
 /**
  * @初始化
  */
 async function init(options) {
+  setTimeout(()=>{
+    if(useStore?.userId){
+      tenantIdChange.value = useStore.tenantId; //切换租户ID
+      console.log("切换租户ID", useStore)
+      getTenantList(useStore.userId); //调用获取企业列表方法
+    }
+  },2000)
   //#ifdef H5
   await useStore.GetWxOpenId(2, options); //调用获取微信公众号openId
   //#endif

+ 2 - 2
src/pages/login.vue

@@ -180,11 +180,11 @@ function init() {
   useStore.GetWxOpenId(1); //调用获取微信公众号openId
 
   if (window.location.host) {
-    // linkUrl.value = window.location.host;
+    linkUrl.value = window.location.host;
     // linkUrl.value = "192.168.10.165:13200";
     // linkUrl.value = "localhost:81";
     // linkUrl.value = "120.26.164.249:13212";
-    linkUrl.value = "manager.usky.cn";
+    // linkUrl.value = "manager.usky.cn";
     useStore.GetMobileTenantConfig({ url: linkUrl.value });
   }
   //#endif

+ 1 - 1
src/plugins/constData.plugins.js

@@ -117,7 +117,7 @@ let projectTabbar = [
 	},
 	{
 		pagePath: "/pages/business/common/projectMange/statistics/index",
-		iconClass: "oaIcon-tab-reportInsert",
+		iconClass: "oaIcon-tab-reportCount",
 		iconPath: "",
 		selectedIconPath: "",
 		text: "报告统计",

+ 72 - 0
src/plugins/jsencrypt.js

@@ -0,0 +1,72 @@
+import JSEncrypt from 'jsencrypt/bin/jsencrypt.min'
+
+// 密钥对生成 http://web.chacuo.net/netrsakeypair
+
+const publicKey = `MIGfMA0GCSqGSIb3DQEBAQUAA4GNADCBiQKBgQCxPWP0HTBE9vEeM34Qx03U8oVm
+C6xIqWPRuI5t8J0zEDQudAgXKPjy8E0Q3cX800UNBTx2gUfRRNrONqALKDnJ1SE6
+qCUDeXOez8sa95GQ9d4BX7pSjZLrPfnCBTBtb5LGkY5zmlmtpG2AV9eJr+kQqhs/
+r0c4njwaDjVG4kF3ZQIDAQAB`
+
+const privateKey = `MIICdwIBADANBgkqhkiG9w0BAQEFAASCAmEwggJdAgEAAoGBALE9Y/QdMET28R4z
+fhDHTdTyhWYLrEipY9G4jm3wnTMQNC50CBco+PLwTRDdxfzTRQ0FPHaBR9FE2s42
+oAsoOcnVITqoJQN5c57Pyxr3kZD13gFfulKNkus9+cIFMG1vksaRjnOaWa2kbYBX
+14mv6RCqGz+vRziePBoONUbiQXdlAgMBAAECgYBjSDdAXEVYrFdeiouYjHwdyAhP
+pERKo5BFvzMRhJIaM353cwnBJ3NkapVQ2Fn6iMIKTB+VZk+7eu1yTAkUluDfLowd
+REZS4ipOBY5UuNnjbXmSOoUQw6vRnox0X4x6S1vd4FBHgpVe1VkiE7Nz5U7Clyd5
+yw2P1lHwMyB/guAH4QJBAN3dGkMASj0jm23maHOfehp/zlACB8HpMKuV4z/bEg45
+nC9Hw5NloUHrXdzEXP1+S46MCH2THflxDVYtnZTRLO0CQQDMgp3Jrn7kkKtNceZF
+R08hLbVmfNlatgONgFJ5JnR+GTQ6o2gwM6SLyoBkfAIiEDpr6c6nBXTU09GOYxBk
++h1ZAkB32pXxVBrG5JF20V3j+GcyIZEGz9H5A0xzpUlambIrVRv2vsH8wo5W2hue
+w8Woe629mBCOJgevVU9rGsFiP44RAkEApbTYAQjAjJakFpZJjKzg8vNEXoye2R9N
+9aOaL8v27A2kAjdRPm050IL+UW0hlVQs4i+KYE7NgX03+PVP3WHD0QJBANLo4PRw
+7Y+dLPAzuazsD3/5SYaSh+KSD/+tVbc6CFvLyfFUKp/a4PzzvGaLo/Ky/ffOY5k0
+hmavbHCKcg+r+hg=`
+
+
+const publicKey2 = `MIGfMA0GCSqGSIb3DQEBAQUAA4GNADCBiQKBgQC9GjiPyz3FfHcEAJu5mgYkaAyt
+xs1kSRIg7j9nFB4cUOKn5flsTwHjn+lR5D0MVc4xS9gtECgOMJsazqsxF35L5TKW
+i0GiX5zw9NhQz6Lv6P30mcm5mJs2UHOyeqr7rtDQn5Uun7Q9cfymQQ+ln7I54rCr
+GhTrkQzMw8+vhTVPKQIDAQAB`
+
+const privateKey2 = `MIICdwIBADANBgkqhkiG9w0BAQEFAASCAmEwggJdAgEAAoGBAL0aOI/LPcV8dwQA
+m7maBiRoDK3GzWRJEiDuP2cUHhxQ4qfl+WxPAeOf6VHkPQxVzjFL2C0QKA4wmxrO
+qzEXfkvlMpaLQaJfnPD02FDPou/o/fSZybmYmzZQc7J6qvuu0NCflS6ftD1x/KZB
+D6WfsjnisKsaFOuRDMzDz6+FNU8pAgMBAAECgYAsJCzD1AttR+p4MrQnFP8ODIO5
+JMjY+f2TVknSg+2ram4eBx67241zVuadyQs/D+B7vVqO2lCfT1VFYqyvXu6J3YX0
+yagkQRiVLJLSMmP14ZCOlVA6jmit0MnMxG0XZltWnjpWyMmlGiNFnmK1Y84b913W
+k1H0mXKnCV2v5xD0AQJBAPQitqPCEPtLECXzdw3DBKNj8w1ldMNHU61dGKF/xDsQ
+6pbX5UZGeHBueqGKB2VqFFMtND6IlEVZ9kVTN1VIYEECQQDGStc+EMn3Xbixk2Al
+cMyfbGtpRQuGUk77uBf0HP7VMoL5j+t8KW3KflgKnY1AJa2VxoCk/Fx1tsdu2CbY
+qLTpAkAZ9PbmQmP70+dLoa8uz2VW2fGlovvfJ15GOHe99A+xGANZNmsyJZv4mEaB
+lmNi7PxsO4oqe+sH1KDSjh57+s4BAkEAp9YK73aobB7AyFT0iVw7ZikPlS+ivJOI
+VNkSNUYhj/TMFU3yxMoQKtfbz0hhmU0K6v30PzF3VQ9bKH/+CV5qmQJBAKhH5m6E
+af/baa6dQJbCCS8aVeiFe0AAUQKamAjkRdfUa1/W6O71A3ePsr41l0X4TGpXPapG
+dc/Ul5Q8wSTXkYE=`
+
+// 加密
+export function encrypt(txt) {
+    const encryptor = new JSEncrypt()
+    encryptor.setPublicKey(publicKey) // 设置公钥
+    return encryptor.encrypt(txt) // 对数据进行加密
+}
+
+// 解密
+export function decrypt(txt) {
+    const encryptor = new JSEncrypt()
+    encryptor.setPrivateKey(privateKey) // 设置私钥
+    return encryptor.decrypt(txt) // 对数据进行解密
+}
+
+// 加密c++业务(詹)
+export function encrypt2(txt) {
+    const encryptor = new JSEncrypt()
+    encryptor.setPublicKey(publicKey2) // 设置公钥
+    return encryptor.encrypt(txt) // 对数据进行加密
+}
+
+// 解密c++业务(詹)
+export function decrypt2(txt) {
+    const encryptor = new JSEncrypt()
+    encryptor.setPrivateKey(privateKey2) // 设置私钥
+    return encryptor.decrypt(txt) // 对数据进行解密
+}

+ 11 - 6
src/static/iconfont/iconfont.css

@@ -1,9 +1,9 @@
+/* 在线链接服务仅供平台体验和调试使用,平台不承诺服务的稳定性,企业客户需下载字体包自行发布使用并做好备份。 */
 @font-face {
-  font-family: "iconfont";
-  /* Project id 4510027 */
-  src: url('https://at.alicdn.com/t/c/font_4510027_0coe61tox5x.woff2?t=1735634817751') format('woff2'),
-    url('https://at.alicdn.com/t/c/font_4510027_0coe61tox5x.woff?t=1735634817751') format('woff'),
-    url('https://at.alicdn.com/t/c/font_4510027_0coe61tox5x.ttf?t=1735634817751') format('truetype');
+  font-family: 'iconfont';  /* Project id 4510027 */
+  src: url('https://at.alicdn.com/t/c/font_4510027_3kicjk51jpr.woff2?t=1740548739368') format('woff2'),
+  url('https://at.alicdn.com/t/c/font_4510027_3kicjk51jpr.woff?t=1740548739368') format('woff'),
+  url('https://at.alicdn.com/t/c/font_4510027_3kicjk51jpr.ttf?t=1740548739368') format('truetype');
 }
 
 .iconfont {
@@ -14,6 +14,10 @@
   -moz-osx-font-smoothing: grayscale;
 }
 
+.oaIcon-tab-reportCount:before {
+  content: "\e624";
+}
+
 .oaIcon-open_door:before {
   content: "\e64e";
 }
@@ -204,4 +208,5 @@
 
 .oaIcon-qingchu:before {
   content: "\e61f";
-}
+}
+

BIN
src/static/iconfont/iconfont.ttf


BIN
src/static/images/projectMange/lock.png


+ 6 - 0
src/store/modules/user.js

@@ -1,6 +1,7 @@
 import { defineStore } from "pinia";
 import { storage, storageSystem } from "@/utils/storage";
 import { getToken, setToken, removeToken } from "@/utils/auth";
+import { encrypt } from "@/plugins/jsencrypt";
 // 接口引用
 import { login, logout, getInfo, getMobileTenantConfig, getCodeImg } from "@/api/login";
 import { getUserProfile, appAdd, appDel, getPageAuthorization } from "@/api/system/user";
@@ -26,6 +27,8 @@ const useStores = defineStore("useStores", {
     avatar: storage.get("avatar"),//用户头像
     roles: storage.get("roles"),//用户权限
     permissions: storage.get("permissions"),
+    userName: "",//用户名
+    password: "",//密码
 
     loginTitle: storage.get("loginTitle"),
     loginBottomTitle: storage.get("loginBottomTitle"),
@@ -50,6 +53,9 @@ const useStores = defineStore("useStores", {
             this.SET_TOKEN(res.data.access_token)
             setToken(res.data.access_token);
             this.codeTime = 0
+            this.userName = encrypt(data.username)
+            this.password = encrypt(data.password)
+            resolve(res);
             resolve();
           })
           .catch((error) => {