ソースを参照

Merge branch 'fanghuisheng' of uskycloud/usky-web-mobile into master

gez 4 ヶ月 前
コミット
d9772c526a
34 ファイル変更1179 行追加290 行削除
  1. 25 0
      src/api/business/door.js
  2. 1 1
      src/api/business/meeting.js
  3. 1 1
      src/config.js
  4. 53 12
      src/pages.json
  5. 1 1
      src/pages/business/common/projectMange/list/addEdit.vue
  6. 5 21
      src/pages/business/common/projectMange/record/index.vue
  7. 2 7
      src/pages/business/common/projectMange/write/index.vue
  8. 1 1
      src/pages/business/common/projectMange/write/insert.vue
  9. 6 10
      src/pages/business/fireIot/alarmManage/alarmDetailsList/index.vue
  10. 18 0
      src/pages/business/fireIot/customManage/index.vue
  11. 2 11
      src/pages/business/fireIot/deviceManage/components/deviceDetailsList.vue
  12. 0 7
      src/pages/business/fireIot/facilitiesManage/facilitiesDetailsList.vue
  13. 16 0
      src/pages/business/fireIot/repairManage/repairDetailsList.vue
  14. 0 0
      src/pages/business/meeting/components/avatarList.vue
  15. 132 0
      src/pages/business/meeting/components/switchSlot.vue
  16. 0 0
      src/pages/business/meeting/components/timeSlot.vue
  17. 1 1
      src/pages/business/meeting/detailed/index.vue
  18. 121 99
      src/pages/business/meeting/index.vue
  19. 251 0
      src/pages/business/meeting/my/index.vue
  20. 0 0
      src/pages/business/meeting/new/index.vue
  21. 5 2
      src/pages/business/zhaf/xunJian/collect/index.vue
  22. 4 1
      src/pages/business/zhaf/xunJian/plan/components/report.vue
  23. 4 1
      src/pages/business/zhaf/xunJian/plan/index.vue
  24. 13 6
      src/pages/common/nfc/index.vue
  25. 22 7
      src/pages/face/api.js
  26. 112 28
      src/pages/face/index.vue
  27. 4 1
      src/plugins/index.js
  28. 56 57
      src/plugins/nfc.plugins.js
  29. 238 0
      src/plugins/permission.plugins.js
  30. BIN
      src/static/face/img/face.gif
  31. BIN
      src/static/face/img/face_detection.gif
  32. 61 10
      src/static/face/index.html
  33. 23 4
      src/utils/hideHead.js
  34. 1 1
      unpackage/config/setting.js

+ 25 - 0
src/api/business/door.js

@@ -0,0 +1,25 @@
+import { request } from "@/utils/request";
+
+/**
+ * 门禁接口集合
+ * @method doorControl 门禁开门
+ * @method doorList 门禁列表
+ */
+export function doorApi() {
+    return {
+        doorControl: (data) => {
+            return request({
+                url: `/service-iot/deviceHttp/control`,
+                method: 'GET',
+                data,
+            });
+        },
+        doorList: (data) => {
+            return request({
+                url: `/service-iot/dmpDeviceInfo/pageWhite`,
+                method: 'POST',
+                data,
+            });
+        },
+    };
+}

+ 1 - 1
src/api/business/meeting.js

@@ -249,7 +249,7 @@ export function signOnOut(data) {
 //门禁开门
 export function control(data) {
     return request({
-        url: `/service-meeting/meetingRoom/502_KAT/223212768/control`,
+        url: `/service-iot/deviceHttp/control`,
         method: 'GET',
         params: data
     })

+ 1 - 1
src/config.js

@@ -2,7 +2,7 @@ import manifest from './manifest.json'
 
 // 应用全局配置
 export default {
-  ip: "192.168.120.165:9300",
+  ip: "192.168.10.165:9300",
 
   //#ifdef APP-PLUS || MP-WEIXIN
   baseUrl: "https://gateWay.usky.cn/prod-api",

+ 53 - 12
src/pages.json

@@ -602,7 +602,12 @@
                     "path": "repairManage/repairDetailsList",
                     "style": {
                         "navigationBarTitleText": "报修列表",
-                        "enablePullDownRefresh": false
+                        "enablePullDownRefresh": false,
+                        "navigationStyle": "custom",
+                        "app-plus": {
+                            "bounce": "none",
+                            "titleNView": false
+                        }
                     }
                 },
                 {
@@ -616,7 +621,12 @@
                     "path": "customManage/index",
                     "style": {
                         "navigationBarTitleText": "客户管理",
-                        "enablePullDownRefresh": false
+                        "enablePullDownRefresh": false,
+                        "navigationStyle": "custom",
+                        "app-plus": {
+                            "bounce": "none",
+                            "titleNView": false
+                        }
                     }
                 },
                 {
@@ -664,7 +674,12 @@
                     "path": "projectMange/record/index",
                     "style": {
                         "navigationBarTitleText": "工作报告",
-                        "enablePullDownRefresh": false
+                        "enablePullDownRefresh": false,
+                        "navigationStyle": "custom",
+                        "app-plus": {
+                            "bounce": "none",
+                            "titleNView": false
+                        }
                     }
                 },
                 {
@@ -707,7 +722,12 @@
                     "path": "projectMange/write/index",
                     "style": {
                         "navigationBarTitleText": "报告填写",
-                        "enablePullDownRefresh": false
+                        "enablePullDownRefresh": false,
+                        "navigationStyle": "custom",
+                        "app-plus": {
+                            "bounce": "none",
+                            "titleNView": false
+                        }
                     }
                 },
                 {
@@ -757,18 +777,27 @@
                             "titleNView": false
                         }
                     }
-                },
+                }
+            ]
+        },
+        {
+            "name": "会议预约系统",
+            "root": "pages/business/meeting/",
+            "pages": [
                 {
-                    "path": "meeting/index",
+                    "path": "index",
                     "style": {
                         "navigationBarTitleText": "会议预约",
-                        "enablePullDownRefresh": false
-                       
-                        
+                        "enablePullDownRefresh": false,
+                        "navigationStyle": "custom",
+                        "app-plus": {
+                            "bounce": "none",
+                            "titleNView": false
+                        }
                     }
                 },
                 {
-                    "path": "meeting/new/index",
+                    "path": "new/index",
                     "style": {
                         "navigationBarTitleText": "新建日程",
                         "enablePullDownRefresh": false,
@@ -780,9 +809,21 @@
                     }
                 },
                 {
-                    "path": "meeting/detailed/index",
+                    "path": "detailed/index",
+                    "style": {
+                        "navigationBarTitleText": "会议室详情",
+                        "enablePullDownRefresh": false,
+                        "navigationStyle": "custom",
+                        "app-plus": {
+                            "bounce": "none",
+                            "titleNView": false
+                        }
+                    }
+                },
+                {
+                    "path": "my/index",
                     "style": {
-                        "navigationBarTitleText": "会议详情",
+                        "navigationBarTitleText": "我的会议",
                         "enablePullDownRefresh": false,
                         "navigationStyle": "custom",
                         "app-plus": {

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

@@ -11,12 +11,12 @@
 
   <oa-scroll
     customClass="record-container scroll-height"
-    :isSticky="false"
     :customStyle="{
       //#ifdef APP-PLUS || MP-WEIXIN
       height: `calc(100vh - (44px + ${proxy.$settingStore.StatusBarHeight}))`,
       //#endif
     }"
+    :isSticky="false"
     :refresherLoad="false"
     :refresherEnabled="false"
     :refresherDefaultStyle="'none'"

+ 5 - 21
src/pages/business/common/projectMange/record/index.vue

@@ -1,12 +1,12 @@
 <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="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>
         </view>
       </template>
-    </u-navbar> -->
+    </u-navbar>
     <u-tabs
       :list="tabsList"
       :current="tabsCurrent"
@@ -53,11 +53,10 @@
 
       <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 reportListNewData" :key="index">
         <view class="content-area-time font14" v-if="!el.id">{{ proxy.$time.jktTimes(el.submitDate, "否") }}</view>
-        <view class="content-area-center bg-white" v-else style="margin:0 10px 10px;border-radius:10px">
+        <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>
@@ -90,8 +89,6 @@
         </view>
       </view>
       <!-- end -->
-
-
     </template>
   </oa-scroll>
 
@@ -204,7 +201,7 @@ const state = reactive({
   tabsCurrent: 0,
 
   loading: false,
-  reportListNewData:[],
+  reportListNewData: [],
   pageSize: 20,
   current: 1,
   total: 0,
@@ -234,7 +231,7 @@ const state = reactive({
   tree: [],
 });
 
-const { tabsList, tabsCurrent,reportListNewData, pageSize, current, total, popup, eventList, modal, timedList, tree } = toRefs(state);
+const { tabsList, tabsCurrent, reportListNewData, pageSize, current, total, popup, eventList, modal, timedList, tree } = toRefs(state);
 /**
  * 操作弹框提醒
  * @param type  弹框类型
@@ -303,12 +300,6 @@ function init() {
       });
 
       state.reportListNewData = arrayList;
-
-
-
-
-
-      
     })
     .catch((err) => {
       state.loading = false;
@@ -438,13 +429,6 @@ onReady(() => {});
 
 onShow(() => {
   state.popup.show = false;
-  //调用系统主题颜色
-  // proxy.$settingStore.systemThemeColor([1]);
-  //设置导航栏颜色
-  uni.setNavigationBarColor({
-     frontColor: '#000000',   //字体颜色
-     backgroundColor: '#ffffff'    //背景颜色
-   })
 });
 
 onLoad((options) => {

+ 2 - 7
src/pages/business/common/projectMange/write/index.vue

@@ -1,14 +1,13 @@
 <template>
-  <!-- <u-sticky class="shadow-default" bgColor="#fff" style="top: 0">
+  <u-sticky class="shadow-default" bgColor="#fff" style="top: 0">
     <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>
         </view>
       </template>
-
     </u-navbar>
-  </u-sticky> -->
+  </u-sticky>
 
   <oa-scroll
     customClass="record-container scroll-height bg-white"
@@ -95,10 +94,6 @@ function goNavigateTo(e) {
 onReady(() => {});
 
 onShow(() => {
-  // uni.setNavigationBarColor({
-  //    frontColor: '#000000',   //字体颜色
-  //    backgroundColor: '#ffffff'    //背景颜色
-  //  })
   //调用系统主题颜色
   proxy.$settingStore.systemThemeColor([1]);
 });

+ 1 - 1
src/pages/business/common/projectMange/write/insert.vue

@@ -109,7 +109,7 @@ function handleSubmit(type, id) {
           ccTo: newData.ccTo,
           reportImage: newData.reportImage,
           reportFile: newData.reportFile,
-          timingTime:newData.timingTime? proxy.$time.getFormatterDate(new Date().getTime()).slice(0,10)+' '+ newData.timingTime.slice(11,20):newData.timingTime,
+          timingTime: newData.timingTime ? proxy.$time.getFormatterDate(new Date().getTime()).slice(0, 10) + " " + newData.timingTime.slice(11, 20) : newData.timingTime,
           isRegularlySend: newData.isRegularlySend,
         };
 

+ 6 - 10
src/pages/business/fireIot/alarmManage/alarmDetailsList/index.vue

@@ -1,12 +1,12 @@
 <template>
   <u-sticky class="shadow-default" bgColor="#fff" style="top: 0">
     <u-navbar :titleStyle="{ color: '#000' }" :autoBack="true" :title="`${state.productName}(${state.total})`" :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>
-    </template>
-  </u-navbar>
+      <template #left>
+        <view class="u-navbar__content__left__item">
+          <u-icon name="arrow-left" size="20" color="#000"></u-icon>
+        </view>
+      </template>
+    </u-navbar>
     <u-tabs
       :list="state.tabsList"
       :current="state.tabsCurrent"
@@ -103,10 +103,6 @@ function selectListApi() {
     if (requset.status === "SUCCESS") {
       state.dataList = requset.data.records;
       state.total = requset.data.total;
-
-      // uni.setNavigationBarTitle({
-      //   title: `${state.productName}(${state.total})`,
-      // });
     }
   });
 }

+ 18 - 0
src/pages/business/fireIot/customManage/index.vue

@@ -1,8 +1,26 @@
 <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">
+      <template #left>
+        <view class="u-navbar__content__left__item">
+          <u-icon name="arrow-left" size="20" color="#000"></u-icon>
+        </view>
+      </template>
+      <template #right>
+        <view class="u-navbar__content__right__item"> </view>
+      </template>
+    </u-navbar>
+  </u-sticky>
+
   <oa-scroll
     customClass="customManage-container scroll-height"
     :pageSize="pageSize"
     :total="total"
+    :customStyle="{
+      //#ifdef APP-PLUS || MP-WEIXIN
+      height: `calc(100vh - (44px + ${proxy.$settingStore.StatusBarHeight}))`,
+      //#endif
+    }"
     :isSticky="false"
     :refresherLoad="true"
     :refresherEnabled="true"

+ 2 - 11
src/pages/business/fireIot/deviceManage/components/deviceDetailsList.vue

@@ -1,5 +1,4 @@
 <template>
-
   <u-navbar :titleStyle="{ color: '#000' }" :autoBack="true" :title="`${productName}(${total})`" :placeholder="true" :safeAreaInsetTop="true" bgColor="#fff">
     <template #left>
       <view class="u-navbar__content__left__item">
@@ -8,7 +7,6 @@
     </template>
   </u-navbar>
 
-
   <oa-scroll
     customClass="scroll-height"
     :pageSize="pageSize"
@@ -113,14 +111,10 @@ function init() {
  */
 
 function dmpDeviceInfoApi() {
-  dmpDeviceInfo({productId: productId.value, deviceName: deviceName.value, current: current.value, size: pageSize.value, deviceStatus:radioValue.value }).then((requset) => {
+  dmpDeviceInfo({ productId: productId.value, deviceName: deviceName.value, current: current.value, size: pageSize.value, deviceStatus: radioValue.value }).then((requset) => {
     if (requset.status == "SUCCESS") {
       dataList.value = requset.data.records;
       total.value = requset.data.total;
-
-      // uni.setNavigationBarTitle({
-      //   title: `${productName.value}(${total.value})`,
-      // });
     }
   });
 }
@@ -142,7 +136,7 @@ function radioChange(e) {
   // console.log(e,'e')
   radioValue.value = e;
   dmpDeviceInfoApi();
-  dropdownShow.value=false
+  dropdownShow.value = false;
 }
 
 /**
@@ -172,9 +166,6 @@ onShow(() => {
 onLoad((options) => {
   if ("productName" in options) {
     productName.value = options.productName;
-    // uni.setNavigationBarTitle({
-    //     title: `${productName.value}(${options.total})`,
-    //   });
   }
   if ("id" in options) {
     productId.value = parseInt(options.id);

+ 0 - 7
src/pages/business/fireIot/facilitiesManage/facilitiesDetailsList.vue

@@ -1,5 +1,4 @@
 <template>
-
   <u-navbar :titleStyle="{ color: '#000' }" :autoBack="true" :title="`${facilityTypeName}(${total})`" :placeholder="true" :safeAreaInsetTop="true" bgColor="#fff">
     <template #left>
       <view class="u-navbar__content__left__item">
@@ -81,12 +80,6 @@ function init() {
 function selectListApi() {
   baseGgpFacility({ facilityType: facilityType.value, facilityName: facilityName.value, current: current.value, size: pageSize.value }).then((requset) => {
     if (requset.status === "SUCCESS") {
-      if (requset.data.records.length > 0) {
-        // uni.setNavigationBarTitle({
-        //   title: `${facilityTypeName.value}(${requset.data.total})`,
-        // });
-      }
-
       dataList.value = requset.data.records;
       total.value = requset.data.total;
     }

+ 16 - 0
src/pages/business/fireIot/repairManage/repairDetailsList.vue

@@ -1,5 +1,16 @@
 <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">
+      <template #left>
+        <view class="u-navbar__content__left__item">
+          <u-icon name="arrow-left" size="20" color="#000"></u-icon>
+        </view>
+      </template>
+      <template #right>
+        <view class="u-navbar__content__right__item"> </view>
+      </template>
+    </u-navbar>
+
     <u-tabs
       :list="tabsList"
       :current="tabsCurrent"
@@ -15,6 +26,11 @@
     customClass="repairManage-container scroll-height"
     :pageSize="pageSize"
     :total="total"
+    :customStyle="{
+      //#ifdef APP-PLUS || MP-WEIXIN
+      height: `calc(100vh - (44px + 44px + ${proxy.$settingStore.StatusBarHeight}))`,
+      //#endif
+    }"
     :isSticky="true"
     :refresherLoad="true"
     :refresherEnabled="true"

+ 0 - 0
src/pages/business/common/meeting/components/avatarList.vue → src/pages/business/meeting/components/avatarList.vue


+ 132 - 0
src/pages/business/meeting/components/switchSlot.vue

@@ -0,0 +1,132 @@
+<template>
+  <view class="flex plr10" style="min-height: 50px; line-height: 20px">
+    <view class="flex mtb-auto ptb10 nav" v-for="item in formData" :key="item" @click="handleClick(item)" style="overflow-x: auto">
+      <view
+        class="radius p5 mr10"
+        :style="{
+          color: form[item.prop] ? activeColor : disabledColor,
+          backgroundColor: `${form[item.prop] ? activeColor : disabledColor}30`,
+        }"
+      >
+        <span v-if="item.type == 'switchTime'">
+          {{ proxy.$dayjs(form[item.prop]).format("MM月DD日") }}
+          {{ weekData[proxy.$dayjs(form[item.prop]).format("d")] }}
+        </span>
+        <span class="flex" v-else>
+          <span class="mr5">{{ form[item.prop] || item.label }}</span>
+          <u-icon name="arrow-down" :color="disabledColor" size="12"> </u-icon>
+        </span>
+      </view>
+    </view>
+    <view class="ml-auto mtb-auto pl10 nav" @click="resetSwitch" :style="{ color: activeColor }">重置</view>
+  </view>
+
+  <u-popup :show="popup.show" mode="bottom" bgColor="#fff" :round="10" :zIndex="10074" @close="popup.show = false">
+    <view
+      :style="{
+        borderTopLeftRadius: '10px',
+        borderTopRightRadius: '10px',
+        overflow: 'hidden',
+      }"
+    >
+      <oa-calendar v-if="popup.type === 'switchTime'" :date="form.date" :showMonth="true" @change="calendarChange" @monthSwitch="calendarMonthSwitch">
+        <template #headerRight> </template>
+      </oa-calendar>
+    </view>
+  </u-popup>
+
+  <u-action-sheet
+    :actions="sheet.actions"
+    :show="sheet.show"
+    cancelText="取消"
+    :round="10"
+    :wrapMaxHeight="'50vh'"
+    :closeOnClickOverlay="true"
+    :safeAreaInsetBottom="true"
+    @close="sheet.show = false"
+    @select="changeSheet"
+  ></u-action-sheet>
+</template>
+<script setup>
+/*----------------------------------依赖引入-----------------------------------*/
+import { onLoad, onShow, onReady, onHide, onLaunch, onNavigationBarButtonTap, onPageScroll } from "@dcloudio/uni-app";
+import { ref, reactive, computed, getCurrentInstance, toRefs, inject } from "vue";
+/*----------------------------------接口引入-----------------------------------*/
+/*----------------------------------组件引入-----------------------------------*/
+/*----------------------------------store引入-----------------------------------*/
+import { useStores, commonStores } from "@/store/modules/index";
+/*----------------------------------公共方法引入-----------------------------------*/
+/*----------------------------------公共变量-----------------------------------*/
+const { proxy } = getCurrentInstance();
+const commonStore = commonStores();
+/*----------------------------------变量声明-----------------------------------*/
+const emit = defineEmits(["handleInit", "onClick", "changeSheet", "reset"]);
+const props = defineProps({
+  form: {
+    type: Object,
+    default: {},
+  }, //选中值
+  formData: {
+    type: Object,
+    default: [],
+  }, //选中值列表
+  weekData: {
+    type: Object,
+    default: ["星期日", "星期一", "星期二", "星期三", "星期四", "星期五", "星期六"],
+  }, //周数据
+  activeColor: {
+    type: String,
+    default: "",
+  }, //选中状态颜色
+  disabledColor: {
+    type: String,
+    default: "",
+  }, //禁用状态颜色
+});
+const state = reactive({
+  sheet: {
+    show: false,
+    title: "",
+    actions: [],
+  },
+  popup: {
+    show: false,
+    type: "",
+    list: {},
+  },
+});
+
+const { form, formData, weekData, activeColor, disabledColor } = toRefs(props);
+const { sheet, popup } = toRefs(state);
+
+function handleClick(event) {
+  if (event.type === "switchTime") {
+    state.popup.show = true;
+    state.popup.type = event.type;
+  } else if (event.type === "select") {
+    state.sheet.show = true;
+    state.sheet.actions = event.data;
+  }
+}
+
+function changeSheet(event) {
+  emit("changeSheet", event);
+}
+
+/** 日历日期改变事件 */
+function calendarChange(e) {
+  props.form.date = e.fulldate;
+  emit("handleInit");
+}
+
+/** 日历月份选择事件 */
+function calendarMonthSwitch(e) {
+  console.log(e);
+}
+
+/** 重置选项卡 */
+function resetSwitch() {
+  emit("reset");
+}
+</script>
+<style lang="scss" scoped></style>

+ 0 - 0
src/pages/business/common/meeting/components/timeSlot.vue → src/pages/business/meeting/components/timeSlot.vue


+ 1 - 1
src/pages/business/common/meeting/detailed/index.vue → src/pages/business/meeting/detailed/index.vue

@@ -1,6 +1,6 @@
 <template>
   <view class="about-container">
-    <u-navbar :titleStyle="{ color: '#000' }" :autoBack="true" title="会议详情" :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>

+ 121 - 99
src/pages/business/common/meeting/index.vue → src/pages/business/meeting/index.vue

@@ -1,5 +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">
+      <template #left>
+        <view class="u-navbar__content__left__item">
+          <u-icon name="arrow-left" size="20" color="#000"></u-icon>
+        </view>
+      </template>
+      <template #right>
+        <view class="u-navbar__content__right__item">
+          <u-icon name="more-dot-fill" size="19" color="#000" @click="handleSheet(true, 'rightMore')"></u-icon>
+        </view>
+      </template>
+    </u-navbar>
 
     <view class="flex plr10">
       <u--input
@@ -14,21 +26,15 @@
       ></u--input>
     </view>
 
-    <view class="flex plr10" style="line-height: 20px">
-      <view class="flex mtb-auto ptb10 nav" style="overflow-x: auto" @click="handlePopup(true, 'switchTime', {})">
-        <view
-          class="radius p5 mr10"
-          :style="{
-            color: proxy.$settingStore.themeColor.color,
-            backgroundColor: `${proxy.$settingStore.themeColor.color}30`,
-          }"
-        >
-          {{ proxy.$dayjs(form.date).format("MM月DD日") }}
-          {{ weekData[proxy.$dayjs(form.date).format("d")] }}
-        </view>
-      </view>
-      <view class="ml-auto mtb-auto pl10 nav" :style="{ color: proxy.$settingStore.themeColor.color }">重置</view>
-    </view>
+    <switchSlot
+      :form="form"
+      :form-data="formData"
+      :activeColor="proxy.$settingStore.themeColor.color"
+      disabledColor="#999999"
+      @changeSheet="1"
+      @reset="resetSwitch()"
+      @handleInit="init()"
+    ></switchSlot>
   </u-sticky>
 
   <oa-scroll
@@ -78,38 +84,33 @@
         overflow: 'hidden',
       }"
     >
-      <oa-calendar v-if="popup.type === 'switchTime'" :date="form.date" :showMonth="true" @change="calendarChange" @monthSwitch="calendarMonthSwitch">
-        <template #headerRight> </template>
-      </oa-calendar>
-
       <view v-if="popup.type === 'aboutMeeting'">
         <view class="flex p15 font16">
           <view class="mtb-auto"> {{ popup.list.roomName }}</view>
-          <view class="mtb-auto ml-auto" @click="sheet.show = true">
+          <view class="mtb-auto ml-auto" @click="handleSheet(true, 'popupMore')">
             <u-icon name="info-circle" size="16" label="更多" labelSize="14" labelColor="#333" color="#333"></u-icon>
           </view>
         </view>
-
         <u-gap height="10" bgColor="#f5f6f7"></u-gap>
-        <u-checkbox-group v-model="checkboxValue" placement="column" shape="circle" :borderBottom="true" @change="checkboxChange" style="height: 42vh; overflow: auto">
-          <view v-for="(item, index) in checkboxList" :key="index">
-            <u-checkbox
-              :customStyle="{ padding: '15px', margin: '0', backgroundColor: `${item.isDisabledColor}30` }"
-              :label="item.timeSlot"
-              :name="item.startTime"
-              :disabled="item.isDisabled || item.isAbout === '已预约'"
-              :activeColor="proxy.$settingStore.themeColor.color"
-            >
-              <template #label>
-                <view class="font15 mtb-auto" :style="{ color: item.isDisabled || item.isAbout === '已预约' ? '#c8c9cc' : '#333333' }"> {{ item.timeSlot }}</view>
-                <view class="font15 mtb-auto ml-auto" :style="{ color: item.isDisabled || item.isAbout === '已预约' ? '#c8c9cc' : '#333333' }"> {{ item.isAbout }}</view>
-              </template>
-            </u-checkbox>
-          </view>
-        </u-checkbox-group>
-
+        <scroll-view scroll-y="true" :scroll-into-view="scrollIntoView" style="height: 42vh">
+          <u-checkbox-group v-model="checkboxValue" placement="column" shape="circle" :borderBottom="true" @change="checkboxChange">
+            <view :id="'item' + item.id" v-for="(item, index) in checkboxList" :key="index">
+              <u-checkbox
+                :customStyle="{ padding: '15px', margin: '0', backgroundColor: `${item.isDisabledColor}30` }"
+                :label="item.timeSlot"
+                :name="item.startTime"
+                :disabled="item.isDisabled || item.isAbout === '已预约'"
+                :activeColor="proxy.$settingStore.themeColor.color"
+              >
+                <template #label>
+                  <view class="font15 mtb-auto" :style="{ color: item.isDisabled || item.isAbout === '已预约' ? '#c8c9cc' : '#333333' }"> {{ item.timeSlot }}</view>
+                  <view class="font15 mtb-auto ml-auto" :style="{ color: item.isDisabled || item.isAbout === '已预约' ? '#c8c9cc' : '#333333' }"> {{ item.isAbout }}</view>
+                </template>
+              </u-checkbox>
+            </view>
+          </u-checkbox-group>
+        </scroll-view>
         <u-gap height="10" bgColor="#f5f6f7"></u-gap>
-
         <view class="p10" style="overflow: hidden">
           <u-button type="primary" @click="handleSubmit('下一步')" text="下一步" style="width: 100px; float: right"></u-button>
         </view>
@@ -126,7 +127,7 @@
     :closeOnClickOverlay="true"
     :safeAreaInsetBottom="true"
     @close="sheet.show = false"
-    @select="handleSheet"
+    @select="changeSheet"
   ></u-action-sheet>
 </template>
 
@@ -135,13 +136,14 @@
 import { onLoad, onShow, onReady, onHide, onLaunch, onUnload, onNavigationBarButtonTap, onPageScroll } from "@dcloudio/uni-app";
 import { ref, reactive, computed, getCurrentInstance, toRefs, inject } from "vue";
 /*----------------------------------接口引入-----------------------------------*/
-import { MeetingRoomList, MeetingRoomReservationList, control } from "@/api/business/meeting.js";
+import { MeetingRoomList, MeetingRoomReservationList } from "@/api/business/meeting.js";
+import { doorApi } from "@/api/business/door.js";
 /*----------------------------------组件引入-----------------------------------*/
 import timeSlot from "./components/timeSlot.vue";
+import switchSlot from "./components/switchSlot.vue";
 /*----------------------------------store引入-----------------------------------*/
 import { useStores, systemStores } from "@/store/modules/index";
 /*----------------------------------公共方法引入-----------------------------------*/
-import { storageSystem } from "@/utils/storage"; // 公共方法引用
 /*----------------------------------公共变量-----------------------------------*/
 const { proxy } = getCurrentInstance();
 const useStore = useStores();
@@ -156,6 +158,15 @@ const state = reactive({
     date: proxy.$dayjs().format("YYYY-MM-DD"),
     meetingRoomName: "",
   },
+  formData: [
+    {
+      type: "switchTime",
+      prop: "date",
+      label: "时间",
+      fontSize: "16",
+      data: [],
+    },
+  ],
   popup: {
     show: false,
     type: "",
@@ -164,30 +175,14 @@ const state = reactive({
   sheet: {
     show: false,
     title: "",
-    actions: [
-      {
-        name: "更多",
-        fontSize: "16",
-        disabled: true,
-      },
-      {
-        name: "门禁开门",
-        fontSize: "16",
-        type: "meeting-room",
-      },
-      {
-        name: "会议室详情",
-        fontSize: "16",
-        type: "meeting-details",
-      },
-    ],
+    actions: [],
   },
-  weekData: ["星期日", "星期一", "星期二", "星期三", "星期四", "星期五", "星期六"],
   checkboxValue: [],
   checkboxList: [],
+  scrollIntoView: "",
 });
 
-const { loading, dataList, pageSize, total, form, popup, sheet, weekData, checkboxValue, checkboxList } = toRefs(state);
+const { loading, dataList, pageSize, total, form, formData, popup, sheet, checkboxValue, checkboxList, scrollIntoView } = toRefs(state);
 
 /** 初始化 */
 function init() {
@@ -209,7 +204,6 @@ function init() {
     .catch((err) => {
       state.loading = false;
     });
-
   state.checkboxList = getAllTimesDay(state.form.date, 30);
 }
 
@@ -228,6 +222,7 @@ function getAllTimesDay(day, min) {
     let timeSlot = proxy.$dayjs(startTime).format("HH:mm") + " - " + proxy.$dayjs(endTime).format("HH:mm"); //获取开始时间和结束时间“时、分”
 
     const array = {
+      id: defaultList.length + 1,
       startTime: startTime,
       endTime: endTime,
       timeSlot: timeSlot,
@@ -242,7 +237,6 @@ function getAllTimesDay(day, min) {
       array.isAbout = "不可预约";
       array.isDisabled = true;
     }
-
     defaultList.push(array);
   }
   return defaultList;
@@ -295,45 +289,32 @@ function handleSubmit() {
     meetingFileList: [], //会议文件信息
   };
 
-  proxy.$tab.navigateTo(`/pages/business/common/meeting/new/index`);
+  proxy.$tab.navigateTo(`/pages/business/meeting/new/index`);
   state.popup.show = false;
 }
 
-/** 实时保存填写数据 */
-function realTimeSaving() {
-  if (!state.form.id) {
-    state.saveTime = proxy.$time.formatterDate(new Date(), "hh:mm");
-    storageSystem.set("project", state);
-  }
-}
-
-/** 日历日期改变事件 */
-function calendarChange(e) {
-  state.form.date = e.fulldate;
-  init();
-}
-
-/** 日历月份选择事件 */
-function calendarMonthSwitch(e) {
-  console.log(e);
-}
-
 /** 操作菜单 */
-function handleSheet(e) {
-  if (e.type == "meeting-details") {
-    proxy.$tab.navigateTo(`/pages/business/common/meeting/detailed/index?roomId=${state.popup.list.roomId}`).then(() => {});
-  } else if (e.type == "meeting-room") {
-    openDoor();
+function changeSheet(e) {
+  if (e.name == "会议室详情") {
+    proxy.$tab.navigateTo(`/pages/business/meeting/detailed/index?roomId=${state.popup.list.roomId}`).then(() => {});
+  } else if (e.name == "门禁开门") {
+    openDoor(e);
+  } else if (e.name == "我的会议") {
+    proxy.$tab.navigateTo(`/pages/business/meeting/my/index`).then(() => {});
   }
 }
 
 /**
  * @门禁开门
  */
-function openDoor() {
-  control({
-    commandStr: JSON.stringify({ method: "control", params: { device_id: "223212768", command: 2 } }),
-  })
+function openDoor(e) {
+  doorApi()
+    .doorControl({
+      productCode: "502_KAT",
+      deviceId: e.doorId,
+      commandCode: "door_onoff",
+      commandValue: 1,
+    })
     .then((item2) => {
       proxy.$modal.msg("开门成功");
     })
@@ -342,6 +323,33 @@ function openDoor() {
     });
 }
 
+/** 操作菜单 */
+function handleSheet(show, type) {
+  state.sheet.show = show;
+  if (type === "rightMore") {
+    state.sheet.actions = [
+      {
+        name: "我的会议",
+        value: "",
+        type: "meeting-my",
+      },
+    ];
+  } else if (type === "popupMore") {
+    state.sheet.actions = [
+      {
+        name: "门禁开门",
+        value: "",
+        type: "meeting-room",
+      },
+      {
+        name: "会议室详情",
+        value: "",
+        type: "meeting-details",
+      },
+    ];
+  }
+}
+
 /** 操作弹窗 */
 function handlePopup(show, type, list) {
   state.popup.show = show;
@@ -362,9 +370,29 @@ function handlePopup(show, type, list) {
         }
       });
     });
+
+    state.scrollIntoView = "";
+    setTimeout(() => {
+      state.checkboxList.some((e) => {
+        if (e.isDisabled === false) {
+          state.scrollIntoView = "item" + e.id;
+          return true;
+        }
+        return false;
+      });
+    });
   }
 }
 
+/** 操作重置 */
+function resetSwitch() {
+  state.form = {
+    date: proxy.$dayjs().format("YYYY-MM-DD"),
+    meetingRoomName: "",
+  };
+  init();
+}
+
 /**
  * @scrollView刷新数据
  */
@@ -375,14 +403,8 @@ function refresh() {
 onReady(() => {});
 
 onShow(() => {
-  //设置导航栏颜色
-  uni.setNavigationBarColor({
-     frontColor: '#000000',   //字体颜色
-     backgroundColor: '#ffffff'    //背景颜色
-   })
   //调用系统主题颜色
-  // proxy.$settingStore.systemThemeColor([1]);
-
+  proxy.$settingStore.systemThemeColor([1]);
   init();
 });
 

+ 251 - 0
src/pages/business/meeting/my/index.vue

@@ -0,0 +1,251 @@
+<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">
+      <template #left>
+        <view class="u-navbar__content__left__item">
+          <u-icon name="arrow-left" size="20" color="#000"></u-icon>
+        </view>
+      </template>
+    </u-navbar>
+
+    <switchSlot
+      :form="form"
+      :form-data="formData"
+      :activeColor="proxy.$settingStore.themeColor.color"
+      disabledColor="#999999"
+      @changeSheet="changeSheet"
+      @reset="resetSwitch()"
+      @handleInit="init()"
+    ></switchSlot>
+  </u-sticky>
+
+  <oa-scroll
+    customClass="record-container scroll-height"
+    :pageSize="form.pageSize"
+    :total="form.total"
+    :isSticky="true"
+    :customStyle="{
+      //#ifdef APP-PLUS || MP-WEIXIN
+      height: `calc(100vh - (94px + ${proxy.$settingStore.StatusBarHeight}))`,
+      //#endif
+      //#ifdef H5
+      height: `calc(100vh - (94px))`,
+      //#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="loading" fontSize="16" style="z-index: 99"></u-loading-page>
+
+      <view class="content-area" v-for="(group, date) in proxy.$common.groupedItems(dataList, 'createTime')" :key="date">
+        <view class="content-area-time font14">{{ proxy.$time.jktTimes(date, "否") }}</view>
+        <view class="content-area-center bg-white radius mb10 mlr10" v-for="(el, ind) in group" :key="ind">
+          <view class="flex mb10">
+            <view class="content-area-center-title font14 mr-auto">方惠圣预约的会议</view>
+            <!-- <u-icon class="" name="more-dot-fill" size="20" color="#000" @click="moreClick(el)"></u-icon> -->
+          </view>
+          <view class="mb5">
+            <view>会议室:{{ el.roomName }}</view>
+            <view>发起人:{{ el.createBy }}</view>
+            <view>参会时间:{{ el.startDate.split("T")[1] }} - {{ el.endDate.split("T")[1] }} </view>
+          </view>
+          <u-divider v-if="el.meetingStatus == 0"></u-divider>
+          <view class="flex">
+            <view class="pb5 mlr-auto text-center" style="width: 100%" @click="handleButton('meeting-release', el)" v-if="el.meetingStatus == 0">释放会议室</view>
+            <!-- <view class="pb5 mlr-auto text-center" style="width: 100%" @click="handleButton('meeting-details', el)">会议详情</view> -->
+          </view>
+        </view>
+      </view>
+    </template>
+  </oa-scroll>
+</template>
+
+<script setup>
+/*----------------------------------依赖引入-----------------------------------*/
+import { onLoad, onShow, onReady, onHide, onLaunch, onUnload, onNavigationBarButtonTap, onPageScroll } from "@dcloudio/uni-app";
+import { ref, reactive, computed, getCurrentInstance, toRefs, inject } from "vue";
+/*----------------------------------接口引入-----------------------------------*/
+import { myApi, recordApi } from "@/api/business/meeting.js";
+/*----------------------------------组件引入-----------------------------------*/
+import switchSlot from "../components/switchSlot.vue";
+/*----------------------------------store引入-----------------------------------*/
+import { useStores, commonStores } from "@/store/modules/index";
+/*----------------------------------公共方法引入-----------------------------------*/
+/*----------------------------------公共变量-----------------------------------*/
+const useStore = useStores();
+const { proxy } = getCurrentInstance();
+const { meeting_type, meeting_status } = proxy.useDict("meeting_type", "meeting_status");
+/*----------------------------------变量声明-----------------------------------*/
+const state = reactive({
+  loading: false,
+  dataList: [],
+  form: {
+    pageSize: 10,
+    current: 1,
+    total: 0,
+    meetingMode: undefined,
+    meetingStatus: undefined,
+  },
+  formData: [
+    {
+      type: "select",
+      prop: "meetingMode",
+      label: "会议类型",
+      fontSize: "16",
+      data: [],
+    },
+    {
+      type: "select",
+      prop: "meetingStatus",
+      label: "会议状态",
+      fontSize: "16",
+      data: [],
+    },
+  ],
+});
+
+const { loading, dataList, form, formData, sheet } = toRefs(state);
+
+/**
+ * @初始化
+ */
+function init() {
+  myApi()
+    .Select({
+      current: state.form.current,
+      size: state.form.pageSize,
+      meetingMode: proxy.$common.mapping("value", "label", state.form.meetingMode, meeting_type.value),
+      meetingStatus: proxy.$common.mapping("value", "label", state.form.meetingStatus, meeting_status.value),
+    })
+    .then((requset) => {
+      state.form.total = requset.data.total;
+      state.dataList = requset.data.records;
+
+      state.formData[0].data = [];
+      state.formData[1].data = [];
+      meeting_type.value.forEach((e) => {
+        state.formData[0].data.push({
+          name: e.label,
+          type: "",
+          prop: state.formData[0].prop,
+          value: e.value,
+        });
+      });
+      meeting_status.value.forEach((e) => {
+        state.formData[1].data.push({
+          name: e.label,
+          type: "",
+          prop: state.formData[1].prop,
+          value: e.value,
+        });
+      });
+    })
+    .catch((err) => {});
+}
+
+/** 操作按钮 */
+function handleButton(type, event) {
+  if (type === "meeting-release") {
+    recordApi()
+      .Cancel(event.meetingId)
+      .then((res) => {
+        proxy.$modal.msg("会议取消");
+        init();
+      });
+  } else if (type === "meeting-details") {
+    // proxy.$tab.navigateTo(`/pages/business/meeting/detailed/index?roomId=${event.roomId}`).then(() => {});
+  }
+}
+
+/** 操作菜单 */
+function changeSheet(e) {
+  state.form[e.prop] = e.name;
+  init();
+}
+
+/** 操作重置 */
+function resetSwitch() {
+  state.form = {
+    pageSize: 10,
+    current: 1,
+    total: 0,
+    meetingMode: undefined,
+    meetingStatus: undefined,
+  };
+  init();
+}
+
+/**
+ * @scrollView加载数据
+ */
+function load() {
+  state.form.pageSize += 10;
+  init();
+}
+
+/**
+ * @scrollView刷新数据
+ */
+function refresh() {
+  state.form.pageSize = 20;
+  init();
+}
+
+onReady(() => {});
+
+onShow(() => {
+  //调用系统主题颜色
+  proxy.$settingStore.systemThemeColor([1]);
+});
+
+onLoad((options) => {
+  init();
+});
+
+onUnload(() => {});
+</script>
+
+<style lang="scss" scoped>
+.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;
+    }
+    &-avatarImg {
+      width: 35px;
+      height: 35px;
+      border-radius: 4px;
+    }
+
+    &-title {
+      margin: 0 0 15px 0;
+      font-weight: 600;
+      color: #000000;
+    }
+  }
+}
+</style>

+ 0 - 0
src/pages/business/common/meeting/new/index.vue → src/pages/business/meeting/new/index.vue


+ 5 - 2
src/pages/business/zhaf/xunJian/collect/index.vue

@@ -97,7 +97,10 @@ const xunJianStore = xunJianStores(); //全局变量值Store
  * @NFC
  */
 function nfcClick() {
-  proxy.$nfc.initNFC();
+  // proxy.$nfc.initNFC();
+  uni.navigateTo({
+    url: "/pages/common/nfc/index",
+  });
 }
 
 /**
@@ -231,7 +234,7 @@ onShow(() => {
     setTimeout(() => {
       proxy.$tab.navigateTo(`/pages/business/zhaf/xunJian/collect/components/collectDetail?siteNubmber=${value}&siteType=${2}`);
       uni.$off("NFC_readID"); //将值删除监听器
-    }, 0);
+    }, 100);
   });
 });
 

+ 4 - 1
src/pages/business/zhaf/xunJian/plan/components/report.vue

@@ -112,7 +112,10 @@ function pulicClick(obj) {
  * @NFC
  */
 function nfcClick() {
-  proxy.$nfc.initNFC();
+  // proxy.$nfc.initNFC();
+  uni.navigateTo({
+    url: "/pages/common/nfc/index",
+  });
 }
 
 /**

+ 4 - 1
src/pages/business/zhaf/xunJian/plan/index.vue

@@ -191,7 +191,10 @@ function reportClick(obj) {
  * @NFC
  */
 function nfcClick() {
-  proxy.$nfc.initNFC();
+  // proxy.$nfc.initNFC();
+  uni.navigateTo({
+    url: "/pages/common/nfc/index",
+  });
 }
 
 /**

+ 13 - 6
src/pages/common/nfc/index.vue

@@ -15,8 +15,20 @@ import { ref, reactive, computed, getCurrentInstance, toRefs, inject, watch } fr
 const { proxy } = getCurrentInstance();
 
 onLoad((options) => {
+  // 开启nfc初始化
+  proxy.$nfc.initNFC();
   // 开启nfc监听
-  proxy.$nfc.readNFC();
+  proxy.$nfc.readNFC().then((event) => {
+    setTimeout(() => {
+      proxy.$tab.navigateBack(1); //返回上一级页面
+      uni.$emit("NFC_readID", event); //将值存储监听器
+    }, 1000);
+  });
+});
+
+onShow(() => {
+  //调用系统主题颜色
+  proxy.$settingStore.systemThemeColor([1]);
 });
 
 onUnload(() => {
@@ -24,11 +36,6 @@ onUnload(() => {
 });
 
 onReady(() => {});
-
-onShow(() => {
-  //调用系统主题颜色
-  proxy.$settingStore.systemThemeColor([1]);
-});
 </script>
 
 <style lang="scss" scoped>

+ 22 - 7
src/pages/face/api.js

@@ -57,11 +57,26 @@ export function signOnOut(data) {
     })
 }
 
-//门禁开门
-export function control(data) {
-    return request({
-        url: `/service-meeting/meetingRoom/502_KAT/223212768/control`,
-        method: 'GET',
-        params: data
-    })
+/**
+ * 门禁接口集合
+ * @method doorControl 门禁开门
+ * @method doorList 门禁列表
+ */
+export function doorApi() {
+    return {
+        doorControl: (data) => {
+            return request({
+                url: `/service-iot/deviceHttp/control`,
+                method: 'GET',
+                data,
+            });
+        },
+        doorList: (data) => {
+            return request({
+                url: `/service-iot/dmpDeviceInfo/pageWhite`,
+                method: 'POST',
+                data,
+            });
+        },
+    };
 }

+ 112 - 28
src/pages/face/index.vue

@@ -29,19 +29,35 @@
       </view>
       <view v-if="meetingRoomList.length > 0">
         <view class="mb10 required">绑定会议室</view>
-        <view>
+        <view class="mb20">
           <u-input
             v-model="form.meetingName"
             placeholder="会议室(必选)"
             suffixIcon="arrow-right"
             suffixIconStyle="color: #909399"
-            border="none"
+            border="bottom"
+            style="padding: 6px 0px"
             disabledColor="transparent"
             disabled
             @click="handlePicker('绑定会议室')"
           />
         </view>
       </view>
+      <view>
+        <view class="mb10 required">绑定门禁</view>
+        <view>
+          <u-input
+            v-model="form.doorName"
+            placeholder="门禁(必选)"
+            suffixIcon="arrow-right"
+            suffixIconStyle="color: #909399"
+            border="none"
+            disabledColor="transparent"
+            disabled
+            @click="handlePicker('绑定门禁')"
+          />
+        </view>
+      </view>
     </view>
   </u-modal>
 
@@ -64,7 +80,7 @@ import config from "@/config";
 import { onLoad, onShow, onReady, onHide, onLaunch, onUnload, onNavigationBarButtonTap, onPageScroll } from "@dcloudio/uni-app";
 import { ref, reactive, computed, getCurrentInstance, toRefs, inject, nextTick, watch } from "vue";
 /*----------------------------------接口引入-----------------------------------*/
-import { meetingApi, faceApi, signOnOut, control } from "./api.js";
+import { meetingApi, faceApi, signOnOut, doorApi } from "./api.js";
 /*----------------------------------组件引入-----------------------------------*/
 /*----------------------------------store引入-----------------------------------*/
 /*----------------------------------公共方法引入-----------------------------------*/
@@ -75,6 +91,7 @@ const state = reactive({
     width: "100%",
     height: "100%",
   },
+  meetingDoorList: [],
   meetingRoomList: [],
   meetingTimeList: [],
   meetingReservaList: {
@@ -100,6 +117,8 @@ const state = reactive({
     domain: undefined,
     meetingId: undefined,
     meetingName: undefined,
+    doorId: undefined,
+    doorName: undefined,
   },
   inter: {
     interMeeting: null,
@@ -109,13 +128,19 @@ const { webviewStyles, meetingRoomList, modal, picker, form, inter } = toRefs(st
 
 // 初始化
 function init() {
-  uni.chooseImage({
-    sourceType: ["camera"],
-    success(res) {
-      console.log(res);
-    },
+  //#ifdef APP-PLUS
+  proxy.$permission.getPermisson("camera").then((res) => {
+    if (res) {
+      handleChildren({
+        funcName: "开启摄像头",
+        data: {},
+      });
+    }
   });
 
+  initNfc();
+  //#endif
+
   var storage = uni.getStorageSync("storage_face");
   if (storage) {
     state.form.domain = storage.domain;
@@ -123,19 +148,32 @@ function init() {
     state.form.port = storage.port ? storage.port : "";
     state.form.meetingId = storage.meetingId || undefined;
     state.form.meetingName = storage.meetingName || undefined;
+    state.form.doorId = storage.doorId || undefined;
+    state.form.doorName = storage.doorName || undefined;
   }
 
-  getMeetingRoomList();
-  getMeetingRoomReservationList();
-  inter.interMeeting = setInterval(() => {
+  if (!inter.interMeeting) {
     getMeetingRoomReservationList();
-  }, 1000 * 5);
+    inter.interMeeting = setInterval(() => {
+      getMeetingRoomReservationList();
+    }, 1000 * 5);
+  }
+}
+
+// 初始化NFC开门
+function initNfc() {
+  proxy.$nfc.initNFC();
+  proxy.$nfc.readNFC().then((e) => {
+    openDoor();
+    initNfc();
+  });
 }
 
 /**
  * @会议室下拉列表
  */
 function getMeetingRoomList() {
+  state.meetingRoomList = [];
   meetingApi()
     .GetMeetingRoomList({
       domain: state.form.domain,
@@ -152,6 +190,31 @@ function getMeetingRoomList() {
     });
 }
 
+/**
+ * @门禁下拉列表
+ */
+function getdoorList() {
+  state.meetingDoorList = [];
+  doorApi()
+    .doorList({
+      current: 1, //页数
+      size: 2000, //条数
+      productCode: "502_KAT", //产品编码
+      deviceStatus: 2, //设备状态;1:在线,2:离线
+      domain: state.form.domain, //域名
+    })
+    .then((requset) => {
+      if (requset.data.records.length > 0) {
+        requset.data.records.forEach((e) => {
+          state.meetingDoorList.push({
+            value: e.deviceId,
+            name: e.deviceName,
+          });
+        });
+      }
+    });
+}
+
 /**
  * @会议室详情列表
  */
@@ -297,7 +360,7 @@ function faceVerify(imageBase) {
         if (state.meetingReservaList.thisVenueData.length > 0) {
           meetingVerify(item);
         } else {
-          openDoor();
+          openDoor(item);
         }
       } else {
         proxy.$modal.msg(item.data.msg);
@@ -309,25 +372,25 @@ function faceVerify(imageBase) {
 /**
  * @会议验证
  */
-function meetingVerify(item) {
+function meetingVerify(event) {
   meetingApi()
     .Attendee({
       domain: state.form.domain,
       meetingId: state.meetingReservaList.thisVenueData[0].meetingId,
-      userId: item.data.userId,
-      userName: item.data.faceName,
+      userId: event.data.userId,
+      userName: event.data.faceName,
     })
     .then((item1) => {
       if (item1.data.status == "1") {
         proxy.$modal.msg(item1.data.msg);
-        state.msg = `[${item.data.faceName}] ${item1.data.msg}`;
+        state.msg = `[${event.data.faceName}] ${item1.data.msg}`;
 
-        openDoor();
+        openDoor(event);
 
         signOnOut({
           domain: state.form.domain,
           meetingId: state.meetingReservaList.thisVenueData[0].meetingId,
-          userId: item.data.userId, //参会人Id
+          userId: event.data.userId, //参会人Id
           mothodType: 0, //签到签退类别(0.签到 1.签退)
           signType: 1, //签到签退方式(0.人工 1.人脸)
         }).then((item2) => {});
@@ -340,11 +403,17 @@ function meetingVerify(item) {
 /**
  * @门禁开门
  */
-function openDoor() {
-  control({
-    domain: state.form.domain,
-    commandStr: JSON.stringify({ method: "control", params: { device_id: "223212768", command: 2 } }),
-  })
+function openDoor(item) {
+  doorApi()
+    .doorControl({
+      domain: state.form.domain,
+      userId: item.data.userId || undefined,
+      userName: item.data.faceName || undefined,
+      productCode: "502_KAT",
+      deviceId: state.form.doorId,
+      commandCode: "door_onoff",
+      commandValue: 1,
+    })
     .then((item2) => {
       proxy.$modal.msg("开门成功");
     })
@@ -372,6 +441,11 @@ function modalConfirm() {
     return;
   }
 
+  if (!state.form.doorName) {
+    proxy.$modal.msg("请选择绑定门禁");
+    return;
+  }
+
   uni.setStorageSync("storage_face", state.form);
   state.modal.show = false;
   getMeetingRoomReservationList();
@@ -385,6 +459,10 @@ function handlePicker(value, index, ind) {
     state.picker.title = "绑定会议室";
     state.picker.list = [state.meetingRoomList];
     state.picker.defaultIndex = 0;
+  } else if (value == "绑定门禁") {
+    state.picker.title = "绑定门禁";
+    state.picker.list = [state.meetingDoorList];
+    state.picker.defaultIndex = 0;
   }
   state.picker.show = true;
 }
@@ -396,6 +474,9 @@ function pickerConfirm(e) {
   if (state.picker.title == "绑定会议室") {
     state.form.meetingId = e.value[0].value;
     state.form.meetingName = e.value[0].name;
+  } else if (state.picker.title == "绑定门禁") {
+    state.form.doorId = e.value[0].value;
+    state.form.doorName = e.value[0].name;
   }
   state.picker.show = false;
 }
@@ -415,7 +496,6 @@ function modalCancel() {
  */
 function analysisData(event) {
   if ("funcName" in event) {
-    console.log(event.funcName);
     if (event.funcName == "打开配置") {
       state.modal.show = true;
     } else if (event.funcName == "人脸识别") {
@@ -454,9 +534,13 @@ window.onmessage = function (event) {
 // #endif
 
 onLoad((options) => {
-  init();
+  setTimeout(() => {
+    init();
+  }, 500);
 });
 
+onShow(() => {});
+
 onUnload(() => {
   clearInterval(inter.interMeeting); //销毁之前定时器
 });
@@ -464,8 +548,6 @@ onUnload(() => {
 watch(
   () => [state.form.linkUrl, state.form.port],
   (val) => {
-    state.meetingRoomList = [];
-
     if (!state.form.linkUrl) {
       return;
     }
@@ -481,8 +563,10 @@ watch(
         domain += ":" + state.form.port;
       }
     }
+
     state.form.domain = domain;
     config.baseUrl = "http://" + state.form.domain + "/prod-api";
+    getdoorList();
     getMeetingRoomList();
   }
 );

+ 4 - 1
src/plugins/index.js

@@ -7,6 +7,7 @@ import time from "./time.plugins.js";
 import constData from "./constData.plugins.js";
 import nfc from "./nfc.plugins.js";
 import keyListen from "./keyListen.plugins.js";
+import permission from "./permission.plugins.js";
 
 import config from "@/config"; // config
 import { useDict } from '@/utils/dict'
@@ -63,6 +64,8 @@ export default {
     // 公共物理按钮监听
     app.provide("$keyListen", keyListen);
     app.config.globalProperties.$keyListen = keyListen;
-
+    // App权限判断
+    app.provide("$permission", permission);
+    app.config.globalProperties.$permission = permission;
   },
 };

+ 56 - 57
src/plugins/nfc.plugins.js

@@ -23,7 +23,7 @@ var techListsArray = [
 export default {
     initNFC() {
         if (uni.getSystemInfoSync().platform == "android") {
-            listenNFCStatus();
+            listenNFCStatus()
         }
 
         if (uni.getSystemInfoSync().platform == "ios") {
@@ -35,9 +35,18 @@ export default {
         }
     },
     readNFC(callback) {
-        if (uni.getSystemInfoSync().platform == "android") {
-            readData(callback);
-        }
+        return new Promise((resolve, reject) => {
+            if (uni.getSystemInfoSync().platform == "android") {
+                plus.globalEvent.addEventListener("newintent", function () {
+                    readyRead = true; //开启读
+                    plus.device.vibrate(500); //调用手机震动
+                    settingStores().nfcWaiting = '请将手机靠近NFC标签'
+                    handleNFCData().then((event) => {
+                        resolve(event)
+                    })
+                }, false);
+            }
+        })
     },
     closeNFC() {
         if (uni.getSystemInfoSync().platform == "android") {
@@ -72,10 +81,6 @@ function listenNFCStatus() {
             return;
         }
 
-        uni.navigateTo({
-            url: "/pages/common/nfc/index",
-        });
-
         var intent = new Intent(main, main.getClass());
         intent.addFlags(Intent.FLAG_ACTIVITY_SINGLE_TOP);
         var pendingIntent = PendingIntent.getActivity(main, 0, intent, 0);
@@ -83,14 +88,6 @@ function listenNFCStatus() {
         ndef.addDataType("*/*");
         var intentFiltersArray = [ndef];
 
-        plus.globalEvent.addEventListener(
-            "newintent",
-            function () {
-                plus.device.vibrate(500);//调用手机震动
-                setTimeout(handleNFCData, 1000);// 轮询调用 NFC
-            },
-            false
-        );
         plus.globalEvent.addEventListener(
             "pause",
             function (e) {
@@ -120,23 +117,23 @@ function listenNFCStatus() {
 }
 
 function handleNFCData() {
-    NdefRecord = plus.android.importClass("android.nfc.NdefRecord");
-    NdefMessage = plus.android.importClass("android.nfc.NdefMessage");
-    var main = plus.android.runtimeMainActivity();
-    var intent = main.getIntent();
-    if ("android.nfc.action.TECH_DISCOVERED" == intent.getAction()) {
-        if (readyWriteData) {
-            //__write(intent);
-            readyWriteData = false;
-        } else if (readyRead) {
-            __read(intent);
-            readyRead = false;
+    return new Promise((resolve, reject) => {
+        NdefRecord = plus.android.importClass("android.nfc.NdefRecord");
+        NdefMessage = plus.android.importClass("android.nfc.NdefMessage");
+        var main = plus.android.runtimeMainActivity();
+        var intent = main.getIntent();
+        if ("android.nfc.action.TECH_DISCOVERED" == intent.getAction()) {
+            if (readyWriteData) {
+                //__write(intent);
+                readyWriteData = false;
+            } else if (readyRead) {
+                __read(intent).then((event) => {
+                    resolve(event)
+                });
+                readyRead = false;
+            }
         }
-    }
-}
-
-function showToast(msg) {
-    plus.nativeUI.toast(msg);
+    })
 }
 
 /**
@@ -209,28 +206,31 @@ function __write(intent) {
  * @returns
  */
 function __read(intent) {
-    try {
-        settingStores().nfcWaiting = '请勿移开标签\n正在读取数据...'
-        // waiting.setTitle("请勿移开标签\n正在读取数据...");
-        var tag = plus.android.importClass("android.nfc.Tag");
-        tag = intent.getParcelableExtra(NfcAdapter.EXTRA_TAG);
-        var bytesId = intent.getByteArrayExtra(NfcAdapter.EXTRA_ID);
-        // waiting.close();
-        var tagid = bytesToHexString(tag.getId());
+    return new Promise((resolve, reject) => {
+        try {
+            settingStores().nfcWaiting = '请勿移开标签\n正在读取数据...'
+            // waiting.setTitle("请勿移开标签\n正在读取数据...");
+            var tag = plus.android.importClass("android.nfc.Tag");
+            tag = intent.getParcelableExtra(NfcAdapter.EXTRA_TAG);
+            var bytesId = intent.getByteArrayExtra(NfcAdapter.EXTRA_ID);
+            // waiting.close();
+            var tagid = bytesToHexString(tag.getId());
 
-        setTimeout(() => {
-            uni.$emit("NFC_readID", tagid); //将值存储监听器
-            tab.navigateBack(1); //返回上一级页面
-            closeReadAndWrite();
-        }, 1000);
-    } catch (e) {
-        uni.showToast({
-            title: e,
-            icon: "none",
-        });
-    }
+            resolve(tagid)
+        } catch (e) {
+            uni.showToast({
+                title: e,
+                icon: "none",
+            });
+        }
+    })
 }
 
+/**
+ * @十六进制转换
+ * @param { 值 } inarray 
+ * @returns 
+ */
 function bytesToHexString(inarray) {
     var i, j, x;
     var hex = ["0", "1", "2", "3", "4", "5", "6", "7", "8", "9", "A", "B", "C", "D", "E", "F"];
@@ -281,14 +281,13 @@ function writeData() {
     // waiting = plus.nativeUI.showWaiting("请将手机靠近NFC标签!");
 }
 
-function readData() {
-    readyRead = true;
-    settingStores().nfcWaiting = '请将手机靠近NFC标签'
-    // waiting = plus.nativeUI.showWaiting("请将手机靠近NFC标签!", {
-    //     modal: false,
-    // });
+function showToast(msg) {
+    plus.nativeUI.toast(msg);
 }
 
+/**
+ * @关闭NFC
+ */
 function closeReadAndWrite() {
     readyWriteData = false;
     readyRead = false;

+ 238 - 0
src/plugins/permission.plugins.js

@@ -0,0 +1,238 @@
+let isIos = false;
+
+// #ifdef APP-PLUS
+isIos = (plus.os.name === 'iOS')
+// #endif
+
+// 判断安卓主方法
+function requestAndroidPermission(permissionID, ifRequest) {
+	return new Promise((resolve, reject) => {
+		plus.android.requestPermissions([permissionID], onSuccess, onError)
+		function onSuccess(res) {
+			const grantedList = res.granted
+			const deniedList = res.deniedPresent
+			const deniedAlwaysList = res.deniedAlways
+			if (grantedList.includes(permissionID)) {
+				resolve(true)
+			} else {
+				resolve(false)
+				ifRequest && gotoAppPermissionSetting()
+			}
+		}
+		function onError(err) {
+			reject(err)
+		}
+	})
+}
+
+// 分别判断Ios
+function judgeIosPermissionPush(ifRequest) {
+	return new Promise(resolve => {
+		var UIApplication = plus.ios.import("UIApplication");
+		var app = UIApplication.sharedApplication();
+		var enabledTypes = 0;
+		if (app.currentUserNotificationSettings) {
+			var settings = app.currentUserNotificationSettings();
+			enabledTypes = settings.plusGetAttribute("types");
+			if (enabledTypes == 0) {
+				ifRequest && gotoAppPermissionSetting()
+				resolve(false)
+			} else {
+				resolve(true)
+			}
+			plus.ios.deleteObject(settings);
+		} else {
+			enabledTypes = app.enabledRemoteNotificationTypes();
+			if (enabledTypes == 0) {
+				ifRequest && gotoAppPermissionSetting()
+				resolve(false)
+			} else {
+				resolve(true)
+			}
+		}
+		plus.ios.deleteObject(app);
+		plus.ios.deleteObject(UIApplication);
+	})
+}
+
+// 判断定位权限是否开启
+function judgeIosPermissionLocation(ifRequest) {
+	return new Promise(resolve => {
+		var cllocationManger = plus.ios.import("CLLocationManager");
+		var status = cllocationManger.authorizationStatus();
+		if (status == 2) {
+			ifRequest && gotoAppPermissionSetting()
+			resolve(false)
+		} else {
+			resolve(true)
+		}
+		plus.ios.deleteObject(cllocationManger);
+	})
+}
+
+// 判断麦克风权限是否开启
+function judgeIosPermissionRecord(ifRequest) {
+	return new Promise(resolve => {
+		var avaudiosession = plus.ios.import("AVAudioSession");
+		var avaudio = avaudiosession.sharedInstance();
+		var permissionStatus = avaudio.recordPermission();
+		if (permissionStatus == 1684369017 || permissionStatus == 1970168948) {
+			ifRequest && gotoAppPermissionSetting()
+			resolve(false)
+		} else {
+			resolve(true)
+		}
+		plus.ios.deleteObject(avaudiosession);
+	})
+}
+
+// 判断相机权限是否开启
+function judgeIosPermissionCamera(ifRequest) {
+	return new Promise(resolve => {
+		var AVCaptureDevice = plus.ios.import("AVCaptureDevice");
+		var authStatus = AVCaptureDevice.authorizationStatusForMediaType('vide');
+		if (authStatus == 3) {
+			resolve(true)
+		} else {
+			ifRequest && gotoAppPermissionSetting()
+			resolve(false)
+		}
+		plus.ios.deleteObject(AVCaptureDevice);
+	})
+}
+
+// 判断相册权限是否开启
+function judgeIosPermissionPhotoLibrary(ifRequest) {
+	return new Promise(resolve => {
+		var PHPhotoLibrary = plus.ios.import("PHPhotoLibrary");
+		var authStatus = PHPhotoLibrary.authorizationStatus();
+		if (authStatus == 3) {
+			resolve(true)
+		} else {
+			ifRequest && gotoAppPermissionSetting()
+			resolve(false)
+		}
+		plus.ios.deleteObject(PHPhotoLibrary);
+	})
+}
+
+// 判断通讯录权限是否开启
+function judgeIosPermissionContact(ifRequest) {
+	return new Promise(resolve => {
+		var CNContactStore = plus.ios.import("CNContactStore");
+		var cnAuthStatus = CNContactStore.authorizationStatusForEntityType(0);
+		if (cnAuthStatus == 3) {
+			resolve(true)
+		} else {
+			ifRequest && gotoAppPermissionSetting()
+			resolve(false)
+		}
+		plus.ios.deleteObject(CNContactStore);
+	})
+}
+
+// 判断所有权限
+function getPermisson(permisson, ifRequest = false) {
+	switch (permisson) {
+		case "location":
+			if (isIos) {
+				return judgeIosPermissionLocation(ifRequest)
+			} else {
+				return requestAndroidPermission("android.permission.ACCESS_FINE_LOCATION", ifRequest)
+			}
+			break;
+		case "camera":
+			if (isIos) {
+				return judgeIosPermissionCamera(ifRequest)
+			} else {
+				return requestAndroidPermission("android.permission.CAMERA", ifRequest)
+			}
+			break;
+		case "photo":
+			if (isIos) {
+				return judgeIosPermissionPhotoLibrary(ifRequest)
+			} else {
+				return requestAndroidPermission("android.permission.READ_EXTERNAL_STORAGE", ifRequest)
+			}
+			break;
+		case "record":
+			if (isIos) {
+				return judgeIosPermissionRecord(ifRequest)
+			} else {
+				return requestAndroidPermission("android.permission.RECORD_AUDIO", ifRequest)
+			}
+			break;
+		case "contact":
+			if (isIos) {
+				return judgeIosPermissionContact(ifRequest)
+			} else {
+				return requestAndroidPermission("android.permission.READ_CONTACTS", ifRequest)
+			}
+			break;
+		case "call":
+			if (isIos) {
+				return Promise.resolve(true)
+			} else {
+				return requestAndroidPermission("android.permission.CALL_PHONE", ifRequest)
+			}
+			break;
+		case "push":
+			if (isIos) {
+				return judgeIosPermissionPush(ifRequest)
+			} else {
+				return Promise.resolve(true)
+			}
+			break;
+		default:
+			break;
+	}
+}
+
+// 去设置
+function gotoAppPermissionSetting() {
+	if (isIos) {
+		var UIApplication = plus.ios.import("UIApplication");
+		var application2 = UIApplication.sharedApplication();
+		var NSURL2 = plus.ios.import("NSURL");
+		// var setting2 = NSURL2.URLWithString("prefs:root=LOCATION_SERVICES");		
+		var setting2 = NSURL2.URLWithString("app-settings:");
+		application2.openURL(setting2);
+
+		plus.ios.deleteObject(setting2);
+		plus.ios.deleteObject(NSURL2);
+		plus.ios.deleteObject(application2);
+	} else {
+		var Intent = plus.android.importClass("android.content.Intent");
+		var Settings = plus.android.importClass("android.provider.Settings");
+		var Uri = plus.android.importClass("android.net.Uri");
+		var mainActivity = plus.android.runtimeMainActivity();
+		var intent = new Intent();
+		intent.setAction(Settings.ACTION_APPLICATION_DETAILS_SETTINGS);
+		var uri = Uri.fromParts("package", mainActivity.getPackageName(), null);
+		intent.setData(uri);
+		mainActivity.startActivity(intent);
+	}
+}
+
+// 检查是否开启了定位
+function checkSystemEnableLocation() {
+	if (isIos) {
+		var result = false;
+		var cllocationManger = plus.ios.import("CLLocationManager");
+		var result = cllocationManger.locationServicesEnabled();
+		plus.ios.deleteObject(cllocationManger);
+		return result;
+	} else {
+		var context = plus.android.importClass("android.content.Context");
+		var locationManager = plus.android.importClass("android.location.LocationManager");
+		var main = plus.android.runtimeMainActivity();
+		var mainSvr = main.getSystemService(context.LOCATION_SERVICE);
+		var result = mainSvr.isProviderEnabled(locationManager.GPS_PROVIDER);
+		return result
+	}
+}
+
+export default {
+	getPermisson,
+	checkSystemEnableLocation
+}

BIN
src/static/face/img/face.gif


BIN
src/static/face/img/face_detection.gif


+ 61 - 10
src/static/face/index.html

@@ -134,6 +134,27 @@
             height: 240px;
         }
 
+        .specialEffects1,
+        .specialEffects2 {
+            width: 445px;
+            height: 330px;
+            position: absolute;
+            z-index: 2000 !important;
+            transform: translate(-50%, -50%);
+            top: 50%;
+            left: 50%;
+        }
+
+        .specialEffects1 {
+            background: url(img/face.gif) no-repeat;
+            background-size: 445px 330px;
+        }
+
+        .specialEffects2 {
+            background: url(img/face_detection.gif) no-repeat;
+            background-size: 445px 330px;
+        }
+
         .home-card-bottom {
             width: 100%;
             position: absolute;
@@ -214,6 +235,9 @@
             <div class="home-card-right-footer" id="home-card-right-footer">
                 <video id="video" width="320" height="240" preload autoplay loop muted></video>
                 <canvas id="canvas" width="320" height="240"></canvas>
+                <!--人脸特效区域-->
+                <div class="specialEffects1" v-if="state.faceImgState"></div>
+                <div class="specialEffects2" v-else></div>
             </div>
 
         </div>
@@ -233,13 +257,16 @@
             props: {},
             data() {
                 return {
+                    tracker: null,
+                    trackerTask: null,
                     state: {
                         dataAll: {},
                         thisVenueData: [],
                         thisVenueTime: {},
                         nextSceneData: [],
                         nextSceneTime: {},
-                        timeList: []
+                        timeList: [],
+                        faceImgState: true,
                     },
                     timeOutEvent: 0
                 };
@@ -258,17 +285,17 @@
                     canvas.style.transform = 'scaleX(-1)';//画布翻转(1.水平翻转-scaleX(-1) 2.垂直翻转-scaleY(-1))
                     var context = canvas.getContext('2d');
                     var time = 5000;
-                    var tracker = new tracking.ObjectTracker('face');
-                    tracker.setInitialScale(4); //设置识别的放大比例
-                    tracker.setStepSize(2);//设置步长
-                    tracker.setEdgesDensity(0.1);//边缘密度
+                    that.tracker = new tracking.ObjectTracker('face');
+                    that.tracker.setInitialScale(4); //设置识别的放大比例
+                    that.tracker.setStepSize(2);//设置步长
+                    that.tracker.setEdgesDensity(0.1);//边缘密度
                     //启动摄像头,并且识别视频内容
-                    var trackerTask = tracking.track('#video', tracker, {
+                    that.trackerTask = tracking.track('#video', that.tracker, {
                         camera: true
                     });
 
                     var flag = true;
-                    tracker.on('track', function (event) {
+                    that.tracker.on('track', function (event) {
                         if (event.data.length === 0) {
                             // console.log('未检测到人脸')
                             context.clearRect(0, 0, canvas.width, canvas.height);
@@ -284,12 +311,14 @@
                             });
                             if (flag) {
                                 console.log("拍照");
+                                that.state.faceImgState = false;
                                 context.drawImage(video, 0, 0, video.width, video.height);
                                 that.saveAsLocalImage();//调用获取图片bold
                                 context.clearRect(0, 0, canvas.width, canvas.height);
                                 flag = false;
                                 setTimeout(function () {
                                     flag = true;
+                                    that.state.faceImgState = true;
                                 }, time);
                             } else {
                                 //console.log("冷却中");
@@ -352,8 +381,8 @@
                             this.state.nextSceneTime = JSON.parse(event.data).nextSceneTime
                             this.state.timeList = JSON.parse(event.data).timeList
                             this.initData();
-                        } else if (event.funcName == "撒点") {
-
+                        } else if (event.funcName == "开启摄像头") {
+                            this.initVido();//调用初始化摄像头
                         }
                     }
                 },
@@ -361,6 +390,28 @@
                 longPress() {
                     this.parentMessage('打开配置')
                     this.timeOutEvent = 0
+                },
+                // 监听页面是否隐藏
+                handleVisibilityChange() {
+                    if (document.visibilityState === 'visible') {
+                        // 页面变为可见时的处理逻辑
+                        console.log('页面变为可见');
+                        this.initVido();
+                    } else if (document.visibilityState === 'hidden') {
+                        // 页面变为不可见时的处理逻辑
+                        console.log('页面变为不可见');
+                        this.closeFace();
+                    }
+                },
+                closeFace() {
+                    try {
+                        this.tracker = null
+                        // 关闭摄像头
+                        let video = document.getElementById('video')
+                        video.srcObject.getTracks()[0].stop()
+                        // 停止侦测
+                        this.trackerTask.stop()
+                    } catch (error) { }
                 }
             },
             created() {
@@ -374,7 +425,7 @@
                 });
             },
             mounted() {
-                this.initVido();//调用初始化摄像头
+                document.addEventListener('visibilitychange', this.handleVisibilityChange);
             },
             beforeDestroy() {
                 // 移除window方法

+ 23 - 4
src/utils/hideHead.js

@@ -1,3 +1,9 @@
+// 页面白名单
+const whiteList = [
+	"pages/index",//登录
+	"pages/info",//消息
+];
+
 export default {
 	mounted() {
 		if (this.isWeiXinBrowser() || this.isQQBrowser()) {
@@ -6,12 +12,14 @@ export default {
 	},
 	methods: {
 		isWeiXinBrowser() { //判断是否为微信
-			let ua = navigator.userAgent.toLowerCase()
-			return ua.indexOf('micromessenger') != -1
+			let ua = navigator?.userAgent.toLowerCase()
+			if (ua) {
+				return ua.indexOf('micromessenger') != -1
+			}
 		},
 		isQQBrowser() { //判断是否为qq
-			var ua = navigator.userAgent.toLowerCase()
-			if (ua.match(/QQ/i) == "qq") {
+			var ua = navigator?.userAgent.toLowerCase()
+			if (ua?.match(/QQ/i) == "qq") {
 				return true
 			} else {
 				return false
@@ -19,10 +27,21 @@ export default {
 		},
 		navTitle() {
 			this.$nextTick(() => {
+				const pages = getCurrentPages(); // 获取当前页面栈
+				const currentPage = pages[pages.length - 1]; // 最后一个元素即为当前页面
+
+				if (whiteList.includes(currentPage?.route)) {
+					return
+				}
+
 				let navTitleDom = document.getElementsByTagName('uni-page-head')
 				if (navTitleDom.length) {
 					navTitleDom[0].style.display = 'none'
 				}
+				let navTitleDom1 = document.querySelector('.u-navbar')
+				if (navTitleDom1) {
+					navTitleDom1.style.display = 'none'
+				}
 			})
 		},
 	}

+ 1 - 1
unpackage/config/setting.js

@@ -27,7 +27,7 @@ filesToModify.forEach((file) => {
                 state.name = '智能会议'
                 state.appid = '__UNI__F3963F8'
                 state.description = '智能会议APP,是一款集成了现代信息技术和智能化管理功能的移动应用程序,旨在提升会议体验和管理效率。'
-                state.versionName = "2.0.1"
+                state.versionName = "2.0.5"
                 state.versionCode = 1
                 state.h5.title = '智能会议'
             }