statistics.vue 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288
  1. <template>
  2. <div style="height: 100%">
  3. <Loading v-show="LOADING"></Loading>
  4. <nav-bar />
  5. <head-info
  6. :nowYearMonth="value"
  7. @getLastMonth="getLastMonth"
  8. @getNextMonth="getNextMonth"
  9. />
  10. <statistics-num :nums="nums" />
  11. <van-divider :style="{ color: '#aaa', borderColor: '#aaa', margin: 0 }" />
  12. <div
  13. ref="calTop"
  14. class="scroll-style"
  15. style="overflow-y: auto"
  16. :style="calRelHeight"
  17. >
  18. <el-calendar v-model="value" :first-day-of-week="7">
  19. <!-- 判断是否属于本月 -->
  20. <template slot="dateCell" slot-scope="{ data }">
  21. <div
  22. class="table-head"
  23. @click="getPunchInfo(formatDay(data.day.split('-')[2]))"
  24. >
  25. {{ formatDay(data.day.split("-")[2]) }}
  26. </div>
  27. <div
  28. class="yc-style"
  29. v-for="(item, index) in statisticsData"
  30. :key="index"
  31. >
  32. <!-- 只显示当月打卡数据 ifNowMonth-->
  33. <template v-if="data.type === 'current-month'">
  34. <template v-if="item.day == formatDay(data.day.split('-')[2])">
  35. <div v-if="item.data.upWorkState === '1'" class="redDot"></div>
  36. <div
  37. v-if="item.data.downWorkState === '1'"
  38. class="blueDot"
  39. :style="{ marginLeft: item.data.upWorkState ? '2px' : 0 }"
  40. ></div>
  41. <div
  42. v-if="
  43. item.data.upWorkState === '2' ||
  44. item.data.downWorkState === '2'
  45. "
  46. class="orangeDot"
  47. :style="{ marginLeft: item.data.downWorkState ? '2px' : 0 }"
  48. ></div>
  49. </template>
  50. </template>
  51. </div>
  52. </template>
  53. </el-calendar>
  54. <van-divider
  55. :style="{ color: '#aaa', borderColor: '#aaa', marginTop: 5 }"
  56. />
  57. <statistics-info :dkInfo="dkInfo" v-if="dkInfo.workDay === 1" />
  58. <van-row class="punch-img-style" v-else-if="dkInfo.workDay === 0">
  59. <van-row class="rest-style">今日休息</van-row>
  60. <van-row>
  61. <van-empty
  62. class="custom-image"
  63. :image="rest"
  64. description="好好休息哦~"
  65. />
  66. </van-row>
  67. </van-row>
  68. <template v-else>
  69. <van-empty
  70. class="custom-image"
  71. :image="empty"
  72. description="暂无任何打卡信息"
  73. />
  74. </template>
  75. </div>
  76. <van-tabbar v-model="active">
  77. <van-tabbar-item to="/punch" icon="location-o">打卡</van-tabbar-item>
  78. <van-tabbar-item icon="records">统计</van-tabbar-item>
  79. </van-tabbar>
  80. </div>
  81. </template>
  82. <script>
  83. import navBar from "@/components/navBarPunch";
  84. import headInfo from "@/components/headInfo";
  85. import statisticsNum from "@/components/statisticsNum";
  86. import statisticsInfo from "@/components/statisticsInfo";
  87. import Loading from "@/components/loading";
  88. import { mapState } from "vuex";
  89. import { statistics, statisticsTop } from "@/api";
  90. import { Notify } from "vant";
  91. import { getYyMmDdTimeStamp } from "@/utils/date";
  92. import rest from "@/assets/rest.png";
  93. import empty from "@/assets/empty.png";
  94. export default {
  95. name: "statistics",
  96. components: {
  97. navBar,
  98. headInfo,
  99. statisticsNum,
  100. statisticsInfo,
  101. Loading
  102. },
  103. data() {
  104. return {
  105. active: 1,
  106. // 图片路径
  107. empty: empty,
  108. rest: rest,
  109. // 滚动高度
  110. calHeight: "",
  111. value: new Date(),
  112. nums: {},
  113. // 统计数据
  114. statisticsData: [],
  115. // 打卡信息
  116. dkInfo: {},
  117. };
  118. },
  119. created() {
  120. this.getStatistics();
  121. },
  122. mounted() {
  123. document.querySelector(".scroll-style .el-calendar__header").remove();
  124. // 获取元素高度
  125. this.$nextTick(() => {
  126. this.calHeight = this.$refs.calTop.getBoundingClientRect().top;
  127. });
  128. },
  129. computed: {
  130. ...mapState(["LOADING"]),
  131. // 计算元素高度
  132. calRelHeight() {
  133. return `height: calc( 100vh - ${this.calHeight}px - 50px )`;
  134. },
  135. // 格式化时间
  136. formatDay() {
  137. return function (time) {
  138. if (time.substring(0, 1) === "0") {
  139. return time.substr(1);
  140. }
  141. return time;
  142. };
  143. },
  144. },
  145. methods: {
  146. // 统计
  147. getStatistics() {
  148. this.$store.commit("showLoading");
  149. let nowMonth = getYyMmDdTimeStamp(this.value);
  150. statistics(nowMonth).then((res) => {
  151. if (res.status === 10000) {
  152. this.statisticsData = res.data;
  153. this.getStatisticsTop(nowMonth);
  154. // 获取当前天的打卡信息
  155. this.getPunchInfo(this.value.getDate());
  156. } else {
  157. Notify({ type: "warning", message: res.message });
  158. this.$store.commit("hideLoading");
  159. }
  160. });
  161. },
  162. // 统计顶部数量
  163. getStatisticsTop(nowMonth) {
  164. statisticsTop(nowMonth).then((res) => {
  165. if (res.status === 10000) {
  166. this.nums = res.data[0] || { early: 0, lack: 0, late: 0 };
  167. this.$store.commit("hideLoading");
  168. } else {
  169. Notify({ type: "warning", message: res.message });
  170. this.$store.commit("hideLoading");
  171. }
  172. });
  173. },
  174. // 获取上一月
  175. getLastMonth() {
  176. this.value = new Date(this.value.setMonth(this.value.getMonth() - 1));
  177. this.getStatistics();
  178. },
  179. // 获取下一月
  180. getNextMonth() {
  181. this.value = new Date(this.value.setMonth(this.value.getMonth() + 1));
  182. this.getStatistics();
  183. },
  184. // 获取打卡信息
  185. getPunchInfo(time) {
  186. let res = this.statisticsData.find((item) => item.day == time);
  187. if(res) {
  188. this.dkInfo = res.data;
  189. } else {
  190. this.dkInfo = {}
  191. }
  192. },
  193. },
  194. };
  195. </script>
  196. <style scoped lang="scss">
  197. $width: 1.2vw;
  198. $height: 0.6vh;
  199. .punch-img-style {
  200. padding-bottom: 2vh;
  201. }
  202. .yc-style {
  203. display: flex;
  204. justify-content: center;
  205. }
  206. .redDot {
  207. width: $width;
  208. height: $height;
  209. border-radius: 50%;
  210. background: #932562;
  211. }
  212. .blueDot {
  213. width: $width;
  214. height: $height;
  215. border-radius: 50%;
  216. background: #0000ff;
  217. }
  218. .orangeDot {
  219. width: $width;
  220. height: $height;
  221. border-radius: 50%;
  222. background: #ff9900;
  223. }
  224. .rest-style {
  225. display: flex;
  226. justify-content: center;
  227. width: 100%;
  228. color: #aaa;
  229. }
  230. .table-head {
  231. height: 100%;
  232. display: flex;
  233. align-items: center;
  234. justify-content: center;
  235. }
  236. /deep/ .el-calendar__body {
  237. padding: 0;
  238. table > thead > th {
  239. text-align: center;
  240. }
  241. }
  242. /deep/ .el-calendar-table tr:first-child td {
  243. border: none;
  244. }
  245. /deep/ .el-calendar-table td {
  246. border: none;
  247. }
  248. /deep/ .el-calendar-table .el-calendar-day {
  249. height: 6vh;
  250. /* width: 14.5vw; */
  251. padding: 1vh 2vw 0 2vw;
  252. /* padding: 0; */
  253. }
  254. /deep/ .el-calendar-table td.is-selected {
  255. background: #fff;
  256. }
  257. /deep/ .el-calendar-table .el-calendar-day:hover {
  258. background: transparent;
  259. }
  260. /deep/ .el-calendar-table td.is-selected .el-calendar-day .table-head {
  261. border-radius: 50%;
  262. background: #67a2ff;
  263. color: #fff;
  264. }
  265. /deep/ .el-calendar-table__row .prev {
  266. pointer-events: none;
  267. }
  268. /deep/ .el-calendar-table__row .next {
  269. pointer-events: none;
  270. }
  271. /deep/ .van-empty__image {
  272. width: 100%;
  273. height: 100%;
  274. }
  275. /deep/ .van-empty {
  276. padding: 0;
  277. }
  278. /deep/ .custom-image .van-empty__image img {
  279. width: 100%;
  280. height: 100%;
  281. }
  282. /deep/ .van-empty__description {
  283. color: #aaa;
  284. }
  285. </style>