lex пре 3 година
родитељ
комит
c1fd189c81

+ 137 - 128
src/api/app.js

@@ -1,223 +1,231 @@
-const axios = require('@/common/axios').default
-import qs from 'qs'
+const axios = require("@/common/axios").default;
+import qs from "qs";
 // import axios from '@/common/axios'
-const api = '/api-teacher'
-import request from '@/helpers/request'
+const api = "/api-teacher";
+import request from "@/helpers/request";
 // 发送登录短信验证码
 const sendSmsRequest = (data) => {
   return request({
-      url: '/code/sendSms',
-      method: 'post',
-      requestType: 'form',
-      data
-  })
-}
+    url: "/code/sendSms",
+    method: "post",
+    requestType: "form",
+    data,
+  });
+};
 // 校验登录图形验证码
 const verifyLoginImage = (data) => {
   return request({
-      url: '/code/verifyLoginImage',
-      method: 'post',
-      requestType: 'form',
-      data
-  })
-}
+    url: "/code/verifyLoginImage",
+    method: "post",
+    requestType: "form",
+    data,
+  });
+};
 
 // 手机号密码方式登录
 const usernameLogin = (data) => {
   return axios({
-      url: '/api-auth/usernameLogin',
-      method: 'post',
-      data: qs.stringify(data)
-  })
-}
+    url: "/api-auth/usernameLogin",
+    method: "post",
+    data: qs.stringify(data),
+  });
+};
 
 // 发送登录短信验证码
 const sendSms = (data) => {
   return axios({
-      url: '/api-student/code/sendSms',
-      method: 'post',
-      data: qs.stringify(data)
-  })
-}
+    url: "/api-student/code/sendSms",
+    method: "post",
+    data: qs.stringify(data),
+  });
+};
 
 // 短信验证码的方式登录
 const smsLogin = (data) => {
   return axios({
-      url: '/api-auth/smsLogin',
-      method: 'post',
-      data: qs.stringify(data)
-  })
-}
+    url: "/api-auth/smsLogin",
+    method: "post",
+    data: qs.stringify(data),
+  });
+};
 
 // 注册登录
 const registerAdd = (data) => {
   return axios({
-      url: '/api-student/register/add',
-      method: 'post',
-      data: qs.stringify(data)
-  })
-}
+    url: "/api-student/register/add",
+    method: "post",
+    data: qs.stringify(data),
+  });
+};
 
 // 设置密码
 const userSetPassword = (data) => {
   return axios({
-      url: '/api-auth/user/setPassword',
-      method: 'post',
-      data: qs.stringify(data)
-  })
-}
+    url: "/api-auth/user/setPassword",
+    method: "post",
+    data: qs.stringify(data),
+  });
+};
 
 // 检查手机号
 const queryUserByPhone = (data) => {
   return axios({
-      url: '/api-auth/user/noAuth/queryUserByPhone',
-      method: 'get',
-      params: data
-  })
-}
+    url: "/api-auth/user/noAuth/queryUserByPhone",
+    method: "get",
+    params: data,
+  });
+};
 
 // 获取课程组详情
 const getGroupDetail = (data) => {
   return axios({
-      url: '/api-student/courseGroup/getGroupDetail',
-      method: 'get',
-      params: data
-  })
-}
+    url: "/api-student/courseGroup/getGroupDetail",
+    method: "get",
+    params: data,
+  });
+};
 
 // 购买课程组
 const buyCourseGroup = (data) => {
   return axios({
-      url: '/api-student/courseGroup/buyCourseGroup',
-      method: 'post',
-      data: qs.stringify(data)
-  })
-}
+    url: "/api-student/courseGroup/buyCourseGroup",
+    method: "post",
+    data: qs.stringify(data),
+  });
+};
 
 // 创建课程
 const createCourseGroup = (data) => {
   return axios({
-    url: api + '/courseGroup/createCourseGroup',
-    method: 'post',
-    data: data
-  })
-}
+    url: api + "/courseGroup/createCourseGroup",
+    method: "post",
+    data: data,
+  });
+};
 
 const sysTenantAccountGet = (data) => {
   return axios({
-    url: api + '/sysTenantAccount/get',
-    method: 'get',
-    params: data
-  })
-}
+    url: api + "/sysTenantAccount/get",
+    method: "get",
+    params: data,
+  });
+};
 
 const queryTenantAccountDetail = (data) => {
   return axios({
-    url: api + '/sysTenantAccount/queryTenantAccountDetail',
-    method: 'get',
-    params: data
-  })
-}
+    url: api + "/sysTenantAccount/queryTenantAccountDetail",
+    method: "get",
+    params: data,
+  });
+};
 
 // 课时使用记录列表
 const teacherCourseMinutes = (data) => {
   return axios({
-    url: api + '/teacherCourseSchedule/teacherCourseMinutes',
-    method: 'get',
-    params: data
-  })
-}
+    url: api + "/teacherCourseSchedule/teacherCourseMinutes",
+    method: "get",
+    params: data,
+  });
+};
 
 // 分页查询活动列表
 const tenantEntryActivitesList = (data) => {
   return axios({
-    url: api + '/tenantEntryActivities/queryPage',
-    method: 'get',
-    params: data
-  })
-}
+    url: api + "/tenantEntryActivities/queryPage",
+    method: "get",
+    params: data,
+  });
+};
 
 // 分页查询活动列表
 const createOrder = (data) => {
   return axios({
-    url: api + '/tenantPaymentOrder/createOrder',
-    method: 'post',
-    data: qs.stringify(data)
-  })
-}
+    url: api + "/tenantPaymentOrder/createOrder",
+    method: "post",
+    data: qs.stringify(data),
+  });
+};
 
 // 订单号查询信息
 const queryByOrderNo = (data) => {
   return axios({
-    url: api + '/tenantPaymentOrder/queryByOrderNo',
-    method: 'get',
-    params: data
-  })
-}
+    url: api + "/tenantPaymentOrder/queryByOrderNo",
+    method: "get",
+    params: data,
+  });
+};
 
 // 分页查询活动列表
 const queryByOrderNoAuth = (data) => {
   return axios({
-    url: api + '/teacherOrder/queryByOrderNo',
-    method: 'get',
-    params: data
-  })
-}
+    url: api + "/teacherOrder/queryByOrderNo",
+    method: "get",
+    params: data,
+  });
+};
 
 // 帮助中心分类
 const helpCenterCatalogList = (data) => {
   return axios({
-    url: '/api-cms/helpCenterCatalog/list',
-    method: 'get',
-    params: data
-  })
-}
+    url: "/api-cms/helpCenterCatalog/list",
+    method: "get",
+    params: data,
+  });
+};
 
 // 帮助中心
 const helpCenterContentList = (data) => {
   return axios({
-    url: '/api-cms/helpCenterContent/list',
-    method: 'get',
-    params: data
-  })
-}
+    url: "/api-cms/helpCenterContent/list",
+    method: "get",
+    params: data,
+  });
+};
 
 // 获取用户基本信息
 const queryUserInfo = (data) => {
   return axios({
-    url: '/api-auth/queryUserInfo',
-    method: 'get',
-    params: data
-  })
-}
+    url: "/api-auth/queryUserInfo",
+    method: "get",
+    params: data,
+  });
+};
 
 // 获取用户基本信息
 const queryTeacherInfo = (data) => {
   return axios({
-    url: api + '/teacher/queryUserInfo',
-    method: 'get',
-    params: data
-  })
-}
+    url: api + "/teacher/queryUserInfo",
+    method: "get",
+    params: data,
+  });
+};
 
 // 获取老师评价和学生评价
 const getStuAndTeaReview = (data) => {
   return axios({
-    url: api + '/courseReview/getStuAndTeaReview',
-    method: 'get',
-    params: data
-  })
-}
+    url: api + "/courseReview/getStuAndTeaReview",
+    method: "get",
+    params: data,
+  });
+};
 
 // 批量添加评价
 const batchAdd = (data) => {
   return axios({
-    url: api + '/courseReview/batchAdd',
-    method: 'post',
-    data: data
-  })
-}
+    url: api + "/courseReview/batchAdd",
+    method: "post",
+    data: data,
+  });
+};
 
+// 上传文件
+const uploadFile = (data) => {
+  return axios({
+    url: "/api-web/uploadFile",
+    method: "post",
+    data: data,
+  });
+};
 
 export {
   sendSmsRequest,
@@ -243,5 +251,6 @@ export {
   queryByOrderNoAuth,
   getStuAndTeaReview,
   batchAdd,
-  queryUserByPhone
-}
+  queryUserByPhone,
+  uploadFile,
+};

+ 200 - 0
src/components/header.vue

@@ -0,0 +1,200 @@
+<template>
+  <div
+    class="mheader"
+    v-if="appHide"
+    :style="{ height: `calc(${titleHeight}px + ${navBarHeight}px)` }"
+  >
+    <header
+      class="m-nav-header"
+      :style="[styleName, headStyle]"
+      :class="[isFixed ? 'fixed' : '', background == 'white' ? 'white' : null]"
+    >
+      <div class="m-nav-bar__left" @click="goBack" v-show="isBack">
+        <van-icon class="arrow-left" name="arrow-left"></van-icon>
+      </div>
+      <div class="m-nav-bar__title">
+        <slot>{{ name ? name : $route.meta.descrition }}</slot>
+      </div>
+      <div class="m-nav-bar__right">
+        <slot name="right"></slot>
+      </div>
+    </header>
+  </div>
+</template>
+
+<script>
+/**
+ * 插槽使用方式
+ *  <template v-slot:right>
+        这里
+    </template>
+ */
+// 判断当前是否是app
+import { postMessage } from "@/helpers/native-message";
+import { browser } from "@/common/common";
+const backStatus = browser().isApp ? false : true;
+export default {
+  name: "mheader",
+  props: {
+    appHide: {
+      name: Boolean,
+      default() {
+        return !backStatus;
+      },
+    },
+    name: String, // 标题名称
+    isBack: {
+      // 是否显示返回按钮
+      type: Boolean,
+      default: backStatus,
+    },
+    isFixed: {
+      // 是否固定顶部
+      type: Boolean,
+      default: true,
+    },
+    styleName: {
+      type: Object,
+      default() {
+        return {};
+      },
+    },
+    background: String, // 背景色
+    titleClass: String, // 标题颜色
+    backUrl: {
+      // 跳转指定的URL
+      type: Object,
+      default: () => {
+        return {
+          callBack: null, // 方法
+          status: false, // 是否用指定的连接跳转
+          path: "", // 跳转路径
+          params: {}, // 跳转参数 目前只能用query的方式
+        };
+      },
+    },
+  },
+  data() {
+    return {
+      title: this.name,
+      backUrlParams: this.backUrl, // 取参数
+      navBarHeight: 0, // 顶部导航栏高度
+      headStyle: {},
+      titleHeight: 44, // 顶部导航高度(默认44px)
+    };
+  },
+  async created() {
+    await postMessage({ api: "setBarStatus", content: { status: 0 } });
+    await postMessage({
+      api: "backIconChange",
+      content: { iconStyle: "black" },
+    });
+    let sNavHeight = sessionStorage.getItem("navHeight");
+    let sTitleHeight = sessionStorage.getItem("titleHeight");
+    if (sNavHeight && sTitleHeight) {
+      this.navBarHeight = sNavHeight;
+      this.headStyle = {
+        paddingTop: sNavHeight + "px",
+        height: sTitleHeight + "px",
+        lineHeight: sTitleHeight + "px",
+      };
+    } else {
+      await postMessage({ api: "getNavHeight" }, (res) => {
+        const { content } = res;
+        let headStyle = {};
+        const dpi = content.dpi || 2;
+        if (content.navHeight) {
+          const navHeight = content.navHeight / dpi;
+          sessionStorage.setItem("navHeight", navHeight);
+          this.navBarHeight = navHeight;
+          headStyle = {
+            paddingTop: navHeight + "px",
+          };
+        }
+        if (content.titleHeight) {
+          // 导航栏的高度
+          const titleHeight = content.titleHeight / dpi;
+          sessionStorage.setItem("titleHeight", titleHeight);
+          this.titleHeight = titleHeight;
+          headStyle.height = titleHeight + "px";
+          headStyle.lineHeight = titleHeight + "px";
+        }
+        this.headStyle = headStyle;
+      });
+    }
+  },
+  methods: {
+    goBack() {
+      // 返回上层
+      let urlObj = this.backUrlParams;
+      // console.log(typeof urlObj.callBack)
+      if (typeof urlObj.callBack == "function") {
+        urlObj.callBack();
+      } else {
+        if (urlObj.status) {
+          this.$router.push({
+            path: urlObj.path,
+            query: urlObj.params,
+          });
+        } else {
+          history.go(-1);
+        }
+      }
+    },
+  },
+};
+</script>
+
+<style lang="less" scoped>
+@import url("../assets/commonLess/variable");
+.mheader {
+  height: 44px;
+  overflow: hidden;
+  transition: all ease 0.2s;
+}
+.m-nav-header {
+  position: absolute;
+  left: 0;
+  top: 0;
+  width: 100%;
+  height: 44px;
+  line-height: 44px;
+  background-color: #fff;
+  color: #000000;
+  text-align: center;
+  user-select: none;
+  &.fixed {
+    position: fixed;
+    z-index: 99;
+  }
+  &.white {
+    background-color: @mColor;
+    color: @whiteColor;
+    .m-nav-bar__title {
+      color: @whiteColor;
+    }
+  }
+  .m-nav-bar__title {
+    max-width: 60%;
+    margin: 0 auto;
+    color: #000000;
+    font-weight: 500;
+    font-size: 0.16rem;
+  }
+  .m-nav-bar__left,
+  .m-nav-bar__right {
+    position: absolute;
+    bottom: 0;
+  }
+  .m-nav-bar__left {
+    left: 0.12rem;
+    .arrow-left {
+      font-size: 0.21rem;
+      vertical-align: middle;
+    }
+  }
+  .m-nav-bar__right {
+    right: 0.12rem;
+  }
+}
+</style>

+ 69 - 30
src/main.js

@@ -1,9 +1,9 @@
-import Vue from 'vue'
-import App from './App.vue'
-import store from './store'
-import router from './router/index'
-import './common/vueFilters'
-import 'babel-polyfill'
+import Vue from "vue";
+import App from "./App.vue";
+import store from "./store";
+import router from "./router/index";
+import "./common/vueFilters";
+import "babel-polyfill";
 import {
   Button,
   Icon,
@@ -49,18 +49,57 @@ import {
   Grid,
   GridItem,
   Calendar,
-  Empty
-} from 'vant'
-Vue.use(Button).use(Icon).use(Tag).use(Swipe).use(SwipeItem)
-  .use(Popup).use(Picker).use(DropdownMenu).use(DropdownItem).use(Search)
-  .use(PullRefresh).use(Toast).use(List).use(Collapse).use(CollapseItem)
-  .use(Tab).use(Tabs).use(Row).use(Col).use(Cell).use(CellGroup)
-  .use(Circle).use(Field).use(DatetimePicker).use(Image).use(Loading)
-  .use(ActionSheet).use(RadioGroup).use(Radio).use(Checkbox).use(CheckboxGroup)
-  .use(CountDown).use(Panel).use(Dialog).use(Sticky).use(Rate).use(Switch).use(ImagePreview).use(NoticeBar)
-  .use(NavBar).use(Divider).use(Grid).use(GridItem).use(Calendar).use(Empty)
+  Empty,
+  Uploader,
+} from "vant";
+Vue.use(Button)
+  .use(Icon)
+  .use(Tag)
+  .use(Swipe)
+  .use(SwipeItem)
+  .use(Popup)
+  .use(Picker)
+  .use(DropdownMenu)
+  .use(DropdownItem)
+  .use(Search)
+  .use(PullRefresh)
+  .use(Toast)
+  .use(List)
+  .use(Collapse)
+  .use(CollapseItem)
+  .use(Tab)
+  .use(Tabs)
+  .use(Row)
+  .use(Col)
+  .use(Cell)
+  .use(CellGroup)
+  .use(Circle)
+  .use(Field)
+  .use(DatetimePicker)
+  .use(Image)
+  .use(Loading)
+  .use(ActionSheet)
+  .use(RadioGroup)
+  .use(Radio)
+  .use(Checkbox)
+  .use(CheckboxGroup)
+  .use(CountDown)
+  .use(Panel)
+  .use(Dialog)
+  .use(Sticky)
+  .use(Rate)
+  .use(Switch)
+  .use(ImagePreview)
+  .use(NoticeBar)
+  .use(NavBar)
+  .use(Divider)
+  .use(Grid)
+  .use(GridItem)
+  .use(Calendar)
+  .use(Empty)
+  .use(Uploader);
 
-Vue.config.productionTip = false
+Vue.config.productionTip = false;
 
 // import Vconsole from 'vconsole'
 // const vconsole = new Vconsole()
@@ -68,22 +107,22 @@ Vue.config.productionTip = false
 Vue.mixin({
   data() {
     return {
-      activeButtonIcon: require('@/assets/images/common/icon_check.png'),
-      inactiveButtonIcon: require('@/assets/images/common/icon_default.png'),
-    }
-  }
-})
+      activeButtonIcon: require("@/assets/images/common/icon_check.png"),
+      inactiveButtonIcon: require("@/assets/images/common/icon_default.png"),
+    };
+  },
+});
 
-import VueAMap from 'vue-amap'
-Vue.use(VueAMap)
+import VueAMap from "vue-amap";
+Vue.use(VueAMap);
 VueAMap.initAMapApiLoader({
-  key: 'c7856e7c812d299cff150e74d60ea608',
-  plugin: ['Geolocation', 'PlaceSearch', 'Geocoder', 'ToolBar'],
-  v: '1.4.4'
-})
+  key: "c7856e7c812d299cff150e74d60ea608",
+  plugin: ["Geolocation", "PlaceSearch", "Geocoder", "ToolBar"],
+  v: "1.4.4",
+});
 
 new Vue({
   store,
   router,
-  render: h => h(App)
-}).$mount('#app')
+  render: (h) => h(App),
+}).$mount("#app");

+ 117 - 82
src/router/index.js

@@ -1,98 +1,133 @@
-import Vue from 'vue'
-import Router from 'vue-router'
-import TeacherRouter from './teacherRouter'
-import AppRouter from './appRouter'
-import AuditionRouter from './auditionRouter'
+import Vue from "vue";
+import Router from "vue-router";
+import TeacherRouter from "./teacherRouter";
+import AppRouter from "./appRouter";
+import AuditionRouter from "./auditionRouter";
 
-
-Vue.use(Router)
+Vue.use(Router);
 
 let defaultRouter = [
   {
-    path: '/',
+    path: "/",
     redirect: {
-      name: 'business'
-    }
-  }, {
-    path: '/queryFortuneBag',
-    name: 'FortuneBag',
-    component: () => import(/* webpackChunkName:'CallNames'*/'@/views/teacher/queryList/queryFortuneBag'),
+      name: "business",
+    },
+  },
+  {
+    path: "/queryFortuneBag",
+    name: "FortuneBag",
+    component: () =>
+      import(/* webpackChunkName:'CallNames'*/ "@/views/teacher/queryList/queryFortuneBag"),
+    meta: {
+      descrition: "页面查看",
+      weight: 1, // 页面权重
+    },
+  },
+  {
+    path: "/queryActiveList",
+    name: "ActiveList",
+    component: () =>
+      import(/* webpackChunkName:'CallNames'*/ "@/views/teacher/queryList/queryActiveList"),
+    meta: {
+      descrition: "页面查看",
+      weight: 1, // 页面权重
+    },
+  },
+  {
+    path: "/guide",
+    name: "guide",
+    component: () =>
+      import(/* webpackChunkName:'CallNames'*/ "@/views/rules/guide"),
+    meta: {
+      descrition: "投屏引导",
+      weight: 1, // 页面权重
+    },
+  },
+  {
+    path: "/applyActive",
+    name: "applyActive",
+    component: () =>
+      import(/* webpackChunkName:'applyActive' */ "@/views/applyActive/index.vue"),
+    meta: {
+      descrition: "考级活动",
+      weight: 0,
+    },
+  },
+  {
+    path: "/registerProtocol",
+    name: "registerProtocol",
+    component: () =>
+      import(/* webpackChunkName:'registerProtocol' */ "@/views/protocol/registerProtocol.vue"),
     meta: {
-      descrition: '页面查看',
-      weight: 1 // 页面权重
-    }
+      descrition: "查看协议",
+      weight: 0,
+    },
   },
   {
-    path: '/queryActiveList',
-    name: 'ActiveList',
-    component: () => import(/* webpackChunkName:'CallNames'*/'@/views/teacher/queryList/queryActiveList'),
+    path: "/auth",
+    name: "auth",
+    component: () =>
+      import(/* webpackChunkName:'auth' */ "@/views/protocol/auth.vue"),
     meta: {
-      descrition: '页面查看',
-      weight: 1 // 页面权重
-    }},
-    {
-      path: '/guide',
-      name: 'guide',
-      component: () => import(/* webpackChunkName:'CallNames'*/'@/views/rules/guide'),
-      meta: {
-        descrition: '投屏引导',
-        weight: 1 // 页面权重
-      }
-    }, {
-      path: '/applyActive',
-      name: 'applyActive',
-      component: () => import(/* webpackChunkName:'applyActive' */'@/views/applyActive/index.vue'),
-      meta: {
-          descrition: '考级活动',
-          weight: 0
-      }
-    }, {
-      path: '/registerProtocol',
-      name: 'registerProtocol',
-      component: () => import(/* webpackChunkName:'registerProtocol' */'@/views/protocol/registerProtocol.vue'),
-      meta: {
-          descrition: '查看协议',
-          weight: 0
-      }
-    },{
-      path: '/auth',
-      name: 'auth',
-      component: () => import(/* webpackChunkName:'auth' */'@/views/protocol/auth.vue'),
-      meta: {
-          descrition: '实名认证',
-          weight: 0
-      }
-    }, {
-      path: '/afterClassEvaluate',
-      name: 'afterClassEvaluate',
-      component: () => import(/* webpackChunkName:'afterClassEvaluate' */'@/views/afterClassEvaluate/index.vue'),
-      meta: {
-          descrition: '课后评价',
-          weight: 0
-      }
-    }, {
-      path: '/afterClassEvaluateDetail',
-      name: 'afterClassEvaluateDetail',
-      component: () => import(/* webpackChunkName:'afterClassEvaluateDetail' */'@/views/afterClassEvaluate/detail.vue'),
-      meta: {
-          descrition: '课后详情',
-          weight: 0
-      }
-    }
-]
+      descrition: "实名认证",
+      weight: 0,
+    },
+  },
+  {
+    path: "/afterClassEvaluate",
+    name: "afterClassEvaluate",
+    component: () =>
+      import(/* webpackChunkName:'afterClassEvaluate' */ "@/views/afterClassEvaluate/index.vue"),
+    meta: {
+      descrition: "课后评价",
+      weight: 0,
+    },
+  },
+  {
+    path: "/afterClassEvaluateDetail",
+    name: "afterClassEvaluateDetail",
+    component: () =>
+      import(/* webpackChunkName:'afterClassEvaluateDetail' */ "@/views/afterClassEvaluate/detail.vue"),
+    meta: {
+      descrition: "课后详情",
+      weight: 0,
+    },
+  },
+  {
+    path: "/massMessage",
+    name: "massMessage",
+    component: () =>
+      import(/* webpackChunkName:'massMessage' */ "@/views/massMessage/index"),
+    meta: {
+      descrition: "群发消息",
+      weight: 2,
+    },
+  },
+  {
+    path: "/massOperation",
+    name: "massOperation",
+    component: () =>
+      import(/* webpackChunkName:'massOperation' */ "@/views/massMessage/operation"),
+    meta: {
+      descrition: "定时消息",
+      weight: 2,
+    },
+  },
+];
 
-defaultRouter = defaultRouter.concat(TeacherRouter)
-                             .concat(AppRouter)
-                             .concat(AuditionRouter)
+defaultRouter = defaultRouter
+  .concat(TeacherRouter)
+  .concat(AppRouter)
+  .concat(AuditionRouter);
 
 const router = new Router({
   // mode: 'history',
   base: process.env.BASE_URL,
   routes: defaultRouter,
-  scrollBehavior () {
-    return { x: 0, y: 0 }
-  }
-})
+  scrollBehavior() {
+    return { x: 0, y: 0 };
+  },
+});
 router.onError((error) => {
   const pattern = /Loading chunk (\d)+ failed/g;
   const isChunkLoadFailed = error.message.match(pattern);
@@ -100,6 +135,6 @@ router.onError((error) => {
   if (isChunkLoadFailed) {
     router.replace(targetPath);
   }
-})
+});
 
-export default router
+export default router;

+ 44 - 0
src/views/massMessage/api.js

@@ -0,0 +1,44 @@
+import request from "@/helpers/request";
+
+export const imSendGroupMessage = (data) => {
+  return request({
+    url: "/imSendGroupMessage/queryPage",
+    method: "post",
+    requestType: "form",
+    data,
+  });
+};
+
+// 查询群列表
+export const queryGroupPage1 = (data) => {
+  return request({
+    url: "/imSendGroupMessage/queryGroupPage1",
+    method: "post",
+    data,
+  });
+};
+
+export const imSendGroupMessageSend = (data) => {
+  return request({
+    url: "/imSendGroupMessage/send",
+    method: "post",
+    data,
+  });
+};
+
+export const imSendGroupMessageDelete = (data) => {
+  return request({
+    url: "/imSendGroupMessage/delete",
+    method: "post",
+    requestType: "form",
+    data,
+  });
+};
+
+export const imSendGroupMessageUpdate = (data) => {
+  return request({
+    url: "/imSendGroupMessage/update",
+    method: "post",
+    data,
+  });
+};

BIN
src/views/massMessage/images/icon_arrow.png


BIN
src/views/massMessage/images/icon_file.png


BIN
src/views/massMessage/images/icon_music.png


BIN
src/views/massMessage/images/icon_timer.png


BIN
src/views/massMessage/images/icon_url.png


BIN
src/views/massMessage/images/icon_vip.png


BIN
src/views/massMessage/images/icon_xly.png


+ 127 - 0
src/views/massMessage/index.vue

@@ -0,0 +1,127 @@
+<template>
+  <div class="massMessage">
+    <van-sticky offset-top="0">
+      <Header :appHide="true" :isFixed="false">
+        <template slot="right">
+          <div @click="onCreate">新建</div>
+        </template>
+      </Header>
+
+      <van-tabs color="#01C1B5" v-model="params.sendFlag" @change="onChange">
+        <van-tab :name="0" title="待发送"></van-tab>
+        <van-tab :name="1" title="已发送"></van-tab>
+      </van-tabs>
+    </van-sticky>
+    <van-list
+      v-model="loading"
+      v-if="dataShow"
+      :finished="finished"
+      finished-text="- 没有更多了 -"
+      :immediate-check="false"
+      @load="getList"
+    >
+      <Item
+        v-for="(item, index) in list"
+        :key="index"
+        :item="item"
+        @onDetail="onDetail"
+      />
+    </van-list>
+    <m-empty v-else msg="暂无数据" />
+  </div>
+</template>
+
+<script>
+import Header from "@/components/header";
+import MEmpty from "@/components/MEmpty";
+import Item from "./item";
+import { imSendGroupMessage } from "./api";
+export default {
+  name: "massMessage",
+  components: { Header, Item, MEmpty },
+  data() {
+    return {
+      list: [],
+      loading: false,
+      finished: false,
+      params: {
+        sendFlag: 0,
+        page: 1,
+        rows: 20,
+      },
+      dataShow: true,
+    };
+  },
+  mounted() {
+    this.getList();
+  },
+  methods: {
+    onCreate() {
+      this.$router.push({
+        path: "/massOperation",
+        query: {
+          type: "create",
+        },
+      });
+    },
+    onDetail(item) {
+      console.log("onDetail", item);
+      // 未发送的可以编辑
+      if (!item.sendFlag) {
+        this.$router.push({
+          path: "/massOperation",
+          query: {
+            type: "update",
+            id: item.id,
+            messageContent: item.messageContent,
+            messageType: item.messageType,
+            fileName: item.fileName,
+            fileUrl: item.fileUrl,
+            sendTime: item.sendTime,
+            sendType: item.sendType,
+            targetIds: item.targetIds,
+          },
+        });
+      }
+    },
+    onChange() {
+      this.list = [];
+      this.params.page = 1;
+      this.dataShow = true;
+      this.loading = true;
+      this.finished = false;
+      this.getList();
+    },
+    async getList() {
+      try {
+        const params = this.params;
+        const res = await imSendGroupMessage(this.params);
+        const result = res.data;
+        this.loading = false;
+        if (params.page > result.pageNo) {
+          return;
+        }
+        params.page = result.pageNo;
+        this.list = this.list.concat(result.rows);
+        if (params.page >= result.totalPage) {
+          this.finished = true;
+        }
+        this.params.page++;
+        // 判断是否有数据
+        if (this.list.length <= 0) {
+          this.dataShow = false;
+        }
+      } catch {
+        this.finished = true;
+        this.dataShow = false;
+      }
+    },
+  },
+};
+</script>
+
+<style lang="less" scoped>
+.massMessage {
+  min-height: 100vh;
+}
+</style>

+ 127 - 0
src/views/massMessage/item.vue

@@ -0,0 +1,127 @@
+<template>
+  <van-cell-group class="cellGroup" @click="onDetail">
+    <!-- {{ JSON.stringify(item) }} -->
+    <van-cell center title-class="titleClass">
+      <template #icon>
+        <img src="./images/icon_timer.png" class="icon_timer" />
+      </template>
+      <template #title>
+        {{ sendTime }}
+      </template>
+      <div class="class">
+        {{ groupIds.length }}个群组
+        <img src="./images/icon_arrow.png" />
+      </div>
+    </van-cell>
+
+    <van-cell>
+      <div class="message_content">发送内容:{{ item.messageContent }}</div>
+      <div class="access" v-if="item.fileUrl">
+        <p class="title">附件</p>
+        <div class="file" v-if="item.messageType == 'FILE'">
+          <img src="./images/icon_file.png" />
+          <span class="van-ellipsis" style="width: 80%">{{
+            item.fileName
+          }}</span>
+        </div>
+        <div class="file" v-if="item.messageType == 'IMG'">
+          <!-- <img src="./images/icon_file.png" /> -->
+          <van-image :src="item.fileUrl" class="img" />
+          <span class="van-ellipsis" style="width: 80%">{{
+            item.fileName
+          }}</span>
+        </div>
+      </div>
+    </van-cell>
+  </van-cell-group>
+</template>
+
+<script>
+import dayjs from "dayjs";
+export default {
+  name: "item",
+  props: ["item"],
+  computed: {
+    groupIds() {
+      const targetIds = this.item.targetIds
+        ? this.item.targetIds.split(",")
+        : [];
+      return targetIds;
+    },
+    sendTime() {
+      return dayjs(this.item.sendTime).format("YYYY-MM-DD HH:mm");
+    },
+  },
+  methods: {
+    onDetail() {
+      this.$listeners.onDetail(this.item);
+    },
+  },
+};
+</script>
+
+<style lang="less" scoped>
+.cellGroup {
+  margin: 0.1rem 0.14rem;
+  border-radius: 0.1rem;
+  overflow: hidden;
+  .titleClass {
+    font-size: 0.14rem;
+    font-weight: 500;
+    color: #333333;
+  }
+}
+.icon_timer {
+  width: 0.16rem;
+  height: 0.16rem;
+  margin-right: 0.08rem;
+}
+.class {
+  display: inline-block;
+  line-height: 0.18rem;
+  background: #fff4f2;
+  border-radius: 0.04rem;
+  padding: 0 0.03rem;
+  font-size: 0.12rem;
+  color: #ff5c00;
+  img {
+    margin-left: 0.03rem;
+    width: 0.05rem;
+    height: 0.07rem;
+  }
+}
+.message_content {
+  font-size: 0.14rem;
+  color: #666666;
+  line-height: 0.28rem;
+}
+.access {
+  padding-top: 0.14rem;
+  padding-bottom: 0.08rem;
+  .title {
+    font-size: 0.14rem;
+    color: #999999;
+    line-height: 0.28rem;
+  }
+  .file {
+    display: flex;
+    align-items: center;
+    background: #f6f8f9;
+    border-radius: 0.06rem;
+    padding: 0.12rem;
+    img {
+      width: 0.22rem;
+      height: 0.26rem;
+      margin-right: 0.1rem;
+    }
+
+    .img {
+      border-radius: 0.06rem;
+      width: 0.4rem;
+      height: 0.4rem;
+      margin-right: 0.1rem;
+      overflow: hidden;
+    }
+  }
+}
+</style>

+ 364 - 0
src/views/massMessage/operation.vue

@@ -0,0 +1,364 @@
+<template>
+  <div class="operation">
+    <Header :appHide="true">
+      <template slot="right">
+        <div v-if="type == 'update'" @click="onDel">删除</div>
+      </template>
+    </Header>
+
+    <van-cell-group class="cellGroup">
+      <van-field
+        label="发送群组"
+        required
+        input-align="right"
+        label-class="labelClass"
+        readonly
+        placeholder="请选择发送群组"
+        is-link
+        @click="groupStatus = true"
+      >
+        <template #input>
+          <div class="class" v-if="selectIds.length > 0">
+            {{ selectIds.length }}个群组
+            <img src="./images/icon_arrow.png" />
+          </div>
+        </template>
+      </van-field>
+      <van-field label="发送方式" input-align="right" required>
+        <template #input>
+          <van-radio-group v-model="form.sendType" direction="horizontal">
+            <van-radio name="NOW">即时发送</van-radio>
+            <van-radio name="TIMING">定时发送</van-radio>
+          </van-radio-group>
+        </template>
+      </van-field>
+      <van-field
+        v-if="form.sendType == 'TIMING'"
+        label="发送时间"
+        required
+        input-align="right"
+        readonly
+        is-link
+        v-model="form.sendTime"
+        placeholder="请选择发送时间"
+        @click="timerStatus = true"
+      ></van-field>
+    </van-cell-group>
+
+    <van-cell-group class="cellGroup">
+      <div
+        class="van-cell--required"
+        style="padding: 10px 16px 0;color: #646566;font-size: 14px"
+      >
+        发送内容
+      </div>
+      <van-field
+        type="textarea"
+        row="2"
+        placeholder="请输入发送内容"
+        :border="false"
+        v-model="form.messageContent"
+      ></van-field>
+      <div class="access">
+        <p class="title">附件</p>
+        <van-field style="padding: 0;" :border="false">
+          <template #input>
+            <van-uploader
+              v-model="uploader"
+              :before-read="beforeRead"
+              :after-read="afterRead"
+              :max-count="1"
+              accept="image/*, .pdf"
+            >
+              <!-- <div class="file">
+                <img src="./images/icon_url.png" class="icon_url" />
+                <div class="uploder-txt">
+                  <p style="color: #1A1A1A; font-size: .14rem;">上传附件</p>
+                  <p class="tips">仅支持图片、PDF,最大上传5M。</p>
+                </div>
+              </div> -->
+            </van-uploader>
+
+            <!-- <template v-if="form.fileUrl">
+              <div class="file" v-if="form.messageType === 'FILE'">
+                <img src="./images/icon_file.png" />
+                <span>{{ form.fileName }}</span>
+              </div>
+              <div class="file" v-if="form.messageType === 'IMG'">
+                <van-image :src="form.fileUrl" class="img" />
+                <span>{{ form.fileName }}</span>
+              </div>
+            </template> -->
+          </template>
+        </van-field>
+        <p class="tips">仅支持图片、PDF。最大上传5M。</p>
+      </div>
+    </van-cell-group>
+
+    <div class="btn-group">
+      <van-button type="primary" round block @click="onSubmit">确定</van-button>
+    </div>
+
+    <van-popup v-model="timerStatus" position="bottom">
+      <van-datetime-picker
+        v-model="date"
+        :formatter="formatter"
+        :min-date="minDate"
+        @cancel="timerStatus = false"
+        @confirm="onConfirm"
+      ></van-datetime-picker>
+    </van-popup>
+
+    <van-popup position="bottom" round v-model="groupStatus">
+      <select-group @onSubmit="onChoise" :selectIds="selectIds" />
+    </van-popup>
+  </div>
+</template>
+
+<script>
+import dayjs from "dayjs";
+import Header from "@/components/header";
+import SelectGroup from "./selectGroup";
+import { uploadFile } from "@/api/app";
+import {
+  imSendGroupMessageSend,
+  imSendGroupMessageDelete,
+  imSendGroupMessageUpdate,
+} from "./api";
+export default {
+  name: "operation",
+  components: { Header, SelectGroup },
+  data() {
+    const query = this.$route.query;
+    // 选择的群组
+    const targetIds = query.targetIds;
+    return {
+      type: query.type || "create",
+      groupStatus: false,
+      timerStatus: false,
+      selectIds: targetIds ? targetIds.split(",") : [],
+      uploader: query.fileUrl ? [{ url: query.fileUrl }] : [],
+      date: null,
+      minDate: new Date(),
+      id: query.id,
+      form: {
+        fileName: query.fileName || "",
+        fileUrl: query.fileUrl || "",
+        messageContent: query.messageContent || "",
+        messageType: query.messageType || "",
+        sendType: query.sendType || "NOW",
+        sendTime: query.sendTime || "",
+        targetIds: targetIds || "",
+      },
+    };
+  },
+  methods: {
+    beforeRead(file) {
+      const isLt2M = file.size / 1024 / 1024 < 5;
+      if (!isLt2M) {
+        this.$toast("上传文件大小不能超过 5MB");
+        return false;
+      }
+      return true;
+    },
+    async afterRead(file) {
+      // 上传头像
+      const form = this.form;
+      try {
+        file.status = "uploading";
+        file.message = "上传中...";
+        let formData = new FormData();
+        formData.append("file", file.file);
+        let res = await uploadFile(formData);
+        console.log(file, res);
+        let result = res.data;
+        if (result.code == 200) {
+          file.status = "done";
+          form.fileName = file.file.name;
+          form.fileUrl = result.data.url;
+          form.messageType = this.onCheckFileType(file.file.type);
+        } else {
+          file.status = "failed";
+          file.message = "上传失败";
+          this.$toast(result.msg);
+          form.fileName = "";
+          form.fileUrl = "";
+          return false;
+        }
+      } catch (err) {
+        file.status = "failed";
+        file.message = "上传失败";
+        form.fileName = "";
+        form.fileUrl = "";
+        return false;
+      }
+    },
+    onCheckFileType(type) {
+      if (type.indexOf("image") != -1) {
+        return "IMG";
+      } else if (type.indexOf("pdf") != -1) {
+        return "FILE";
+      }
+      return "";
+    },
+    onDel() {
+      this.$dialog
+        .confirm({
+          title: "提示",
+          message: "确定删除吗?",
+          confirmButtonColor: "#01C1B5",
+        })
+        .then(async () => {
+          try {
+            await imSendGroupMessageDelete({
+              id: this.id,
+            });
+            this.$toast.success("删除成功");
+            setTimeout(() => {
+              this.$router.back();
+            }, 1000);
+          } catch {
+            //
+          }
+        });
+    },
+    formatter(type, val) {
+      if (type === "year") {
+        return `${val}年`;
+      } else if (type === "month") {
+        return `${val}月`;
+      } else if (type === "day") {
+        return `${val}日`;
+      } else if (type === "hour") {
+        return `${val}时`;
+      } else if (type === "minute") {
+        return `${val}分`;
+      }
+      return val;
+    },
+    onChoise(item) {
+      this.groupStatus = false;
+      this.selectIds = item;
+    },
+    onConfirm(date) {
+      this.form.sendTime = dayjs(date).format("YYYY-MM-DD HH:mm:ss");
+      this.timerStatus = false;
+    },
+    async onSubmit() {
+      try {
+        const form = this.form;
+        if (this.selectIds && this.selectIds.length <= 0) {
+          this.$toast("请选择发送群组");
+          return;
+        }
+        if (form.sendType == "TIMING" && !form.sendTime) {
+          this.$toast("请选择发送时间");
+          return;
+        }
+        if (!form.messageContent) {
+          this.$toast("请输入消息内容");
+          return;
+        }
+        form.targetIds = this.selectIds.join(",");
+        if (this.type == "create") {
+          await imSendGroupMessageSend(form);
+          this.$toast.success("创建成功");
+        } else {
+          form.id = this.id;
+          await imSendGroupMessageUpdate(form);
+          this.$toast.success("修改成功");
+        }
+        setTimeout(() => {
+          this.$router.back();
+        }, 1000);
+      } catch {
+        //
+      }
+    },
+  },
+};
+</script>
+
+<style lang="less" scoped>
+.operation {
+  min-height: 100vh;
+}
+.cellGroup {
+  margin: 0.1rem 0.14rem;
+  border-radius: 0.1rem;
+  overflow: hidden;
+  .labelClass {
+    font-size: 0.14rem;
+    font-weight: 500;
+    color: #333333;
+  }
+}
+.class {
+  display: flex;
+  align-items: center;
+  line-height: 0.18rem;
+  background: #fff4f2;
+  border-radius: 0.04rem;
+  padding: 0 0.03rem;
+  font-size: 0.12rem;
+  color: #ff5c00;
+  img {
+    margin-left: 0.03rem;
+    width: 0.05rem;
+    height: 0.07rem;
+  }
+}
+.access {
+  padding: 0 16px 0.2rem;
+  .title {
+    font-size: 0.14rem;
+    color: #999999;
+    line-height: 0.28rem;
+  }
+  .file {
+    display: flex;
+    align-items: center;
+    background: #f6f8f9;
+    border-radius: 0.06rem;
+    padding: 0.12rem;
+    img {
+      width: 0.22rem;
+      height: 0.26rem;
+      margin-right: 0.1rem;
+    }
+
+    .img {
+      border-radius: 0.06rem;
+      width: 0.4rem;
+      height: 0.4rem;
+      margin-right: 0.1rem;
+      overflow: hidden;
+    }
+
+    .icon_url {
+      width: 0.24rem;
+      height: 0.29rem;
+      margin-right: 0.1rem;
+      overflow: hidden;
+    }
+  }
+}
+.tips {
+  font-size: 0.13rem;
+  color: #999999;
+}
+// /deep/.van-uploader {
+//   width: 100%;
+// }
+// /deep/.van-uploader__wrapper {
+//   display: block;
+// }
+.uploder-txt {
+  p {
+    line-height: 1.2;
+  }
+}
+.btn-group {
+  padding: 0 0.14rem;
+}
+</style>

+ 161 - 0
src/views/massMessage/selectGroup.vue

@@ -0,0 +1,161 @@
+<template>
+  <div class="selectGroup">
+    <m-search @onSearch="onSearch" />
+
+    <div class="group_section">
+      <van-list
+        v-model="loading"
+        v-if="dataShow"
+        :finished="finished"
+        finished-text=" "
+        :immediate-check="false"
+        @load="getList"
+      >
+        <van-checkbox-group v-model="groupIds">
+          <van-cell
+            v-for="(item, index) in list"
+            :key="index"
+            @click="onCheckbox(item)"
+          >
+            <template #icon>
+              <van-checkbox :name="item.id"></van-checkbox>
+              <!-- TRAINING MUSIC  VIP/COMM -->
+              <img
+                v-if="item.type === 'MUSIC'"
+                src="./images/icon_music.png"
+                class="group_icon"
+              />
+              <img
+                v-if="item.type === 'TRAINING'"
+                src="./images/icon_xly.png"
+                class="group_icon"
+              />
+              <img
+                v-if="item.type === 'VIP' || item.type === 'COMM'"
+                src="./images/icon_vip.png"
+                class="group_icon"
+              />
+            </template>
+            <div class="group_content">
+              <div class="group_title van-ellipsis">{{ item.name }}</div>
+              <div class="group_desc van-ellipsis">
+                {{ item.memo }}(共{{ item.memberNum || 0 }}人)
+              </div>
+            </div>
+          </van-cell>
+        </van-checkbox-group>
+      </van-list>
+      <m-empty v-else msg="暂无数据" />
+    </div>
+    <div class="btn-group">
+      <van-button type="primary" @click="onSubmit" round block>
+        确认
+      </van-button>
+    </div>
+  </div>
+</template>
+
+<script>
+import MEmpty from "@/components/MEmpty";
+import { queryGroupPage1 } from "./api";
+import MSearch from "@/components/Search";
+export default {
+  name: "selectGroup",
+  props: {
+    selectIds: {
+      type: Array,
+      default() {
+        return [];
+      },
+    },
+  },
+  components: {
+    MSearch,
+    MEmpty,
+  },
+  data() {
+    return {
+      list: [],
+      groupIds: [],
+      loading: false,
+      finished: false,
+      params: {
+        search: "",
+        active: 0,
+        page: 1,
+        rows: 20,
+      },
+      dataShow: true,
+    };
+  },
+  mounted() {
+    // 初始化选择的群组
+    this.groupIds = [...this.selectIds];
+    this.getList();
+  },
+  methods: {
+    onSearch(val) {
+      this.dataShow = true;
+      this.loading = false;
+      this.finished = false;
+      this.params.search = val;
+      this.params.page = 1;
+      this.list = [];
+      this.getList();
+    },
+    onSubmit() {
+      // 确定选择的群组
+      console.log(this.groupIds);
+      this.$emit("onSubmit", this.groupIds);
+    },
+    onCheckbox(item) {
+      const index = this.groupIds.indexOf(item.id);
+      if (index > -1) {
+        this.groupIds.splice(index, 1);
+      } else {
+        this.groupIds.push(item.id);
+      }
+    },
+    async getList() {
+      try {
+        const params = this.params;
+        const res = await queryGroupPage1(params);
+        const result = res.data;
+        this.loading = false;
+        if (params.page > result.pageNo) {
+          return;
+        }
+        params.page = result.pageNo;
+        this.list = this.list.concat(result.rows);
+        if (params.page >= result.totalPage) {
+          this.finished = true;
+        }
+        this.params.page++;
+        // 判断是否有数据
+        if (this.list.length <= 0) {
+          this.dataShow = false;
+        }
+      } catch {
+        this.finished = true;
+        this.dataShow = false;
+      }
+    },
+  },
+};
+</script>
+
+<style lang="less" scoped>
+.group_section {
+  height: 65vh;
+  overflow: auto;
+}
+.group_icon {
+  width: 0.42rem;
+  height: 0.42rem;
+  margin-right: 0.15rem;
+  margin-left: 0.15rem;
+}
+.btn-group {
+  padding: 0.1rem 0.14rem;
+}
+</style>