index.vue 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504
  1. <template>
  2. <u-sticky class="shadow-default" bgColor="#fff" style="top: 0">
  3. <u-navbar :titleStyle="{ color: '#000' }" :autoBack="true" title="工作报告" :placeholder="true" :safeAreaInsetTop="true" bgColor="#fff">
  4. <template #left>
  5. <view class="u-navbar__content__left__item">
  6. <u-icon name="arrow-left" size="20" color="#000"></u-icon>
  7. </view>
  8. </template>
  9. </u-navbar>
  10. <u-tabs
  11. :list="tabsList"
  12. :current="tabsCurrent"
  13. @click="tabsClick"
  14. lineColor="#333"
  15. :activeStyle="{ color: '#333', fontSize: '14px' }"
  16. :inactiveStyle="{ color: '#909399', fontSize: '14px' }"
  17. :scrollable="false"
  18. ></u-tabs>
  19. </u-sticky>
  20. <oa-scroll
  21. customClass="record-container scroll-height"
  22. :pageSize="pageSize"
  23. :total="total"
  24. :isSticky="true"
  25. :customStyle="{
  26. //#ifdef APP-PLUS || MP-WEIXIN
  27. height: `calc(100vh - (138px + ${proxy.$settingStore.StatusBarHeight}))`,
  28. //#endif
  29. //#ifdef H5
  30. height: `calc(100vh - (138px))`,
  31. //#endif
  32. }"
  33. :refresherLoad="true"
  34. :refresherEnabled="true"
  35. :refresherDefaultStyle="'none'"
  36. :refresherThreshold="44"
  37. :lowerThreshold="44"
  38. :refresherBackground="'#f5f6f7'"
  39. @load="load"
  40. @refresh="refresh"
  41. :data-theme="'theme-' + proxy.$settingStore.themeColor.name"
  42. >
  43. <template #default>
  44. <view v-if="timedList.length > 0" @click="goTimingPage()" class="list-cell list-cell-arrow" style="margin: 20px 0; color: rgb(20, 158, 255); background-color: rgba(20, 158, 255, 0.1)">
  45. <view style="width: calc(100% - 51px); display: flex; padding-right: 10px">
  46. <u-icon name="info-circle" color="#2979ff" size="18" style="margin-right: 5px"></u-icon>
  47. <view v-if="timedList.length == 1">您有1条定时日志将于{{ timedList[0].timingTime.slice(0, 16) }}发布</view>
  48. <view v-else>您有{{ timedList.length }}条定时日志将于指定时间发布</view>
  49. </view>
  50. </view>
  51. <u-loading-page :loading="state.loading" fontSize="16" style="z-index: 99"></u-loading-page>
  52. <!-- start -->
  53. <view class="content-area" v-for="(el, index) in reportListNewData" :key="index">
  54. <view class="content-area-time font14" v-if="!el.id">{{ proxy.$time.jktTimes(el.submitDate, "否") }}</view>
  55. <view class="content-area-center bg-white" v-else style="margin: 0 10px 10px; border-radius: 10px">
  56. <view class="content-area-top menu-item" style="float: right; padding: 10px 0px">
  57. <view class="content-area-top-time"> </view>
  58. <u-icon class="content-area-top-icon" name="more-dot-fill" size="20" color="#000" @click="moreClick(el)"></u-icon>
  59. </view>
  60. <view class="flex mb10" @click="goContentDetails(el)">
  61. <img :src="el.avatar" class="content-area-center-avatarImg mr10" v-if="el.avatar" />
  62. <u-avatar
  63. v-if="!el.avatar"
  64. class="content-area-center-avatar mr10"
  65. :text="el.createBy.length > 2 ? el.createBy.slice(1, 3) : el.createBy"
  66. shape="square"
  67. size="35"
  68. fontSize="10"
  69. color="#ffffff"
  70. :bgColor="proxy.$settingStore.themeColor.color"
  71. ></u-avatar>
  72. <view>
  73. <view class="content-area-center-title font14 mb5">{{ el.createBy }}的日报</view>
  74. <view class="content-area-center-time font12">{{ proxy.$time.jktTimes(el.submitDate.replace("T", " ")) }}</view>
  75. </view>
  76. </view>
  77. <view class="mb5" @click="goContentDetails(el)">
  78. <u-text :text="el.contentText.length >= 100 ? el.contentText.slice(0, 100) + '···' : el.contentText" color="#666666" size="14"></u-text>
  79. </view>
  80. <view class="flex" v-if="el.createBy != useStore.nickName" @click="goContentDetails(el)">
  81. <u-tag class="mr10" type="info" text="已读" size="mini" plain v-if="el.readFlag === 1" style="margin: 0 auto"></u-tag>
  82. <u-tag class="mr10" type="error" text="未读" size="mini" plain v-if="el.readFlag === 0" style="margin: 0 auto"></u-tag>
  83. <u-text text="全文" :color="proxy.$settingStore.themeColor.color" size="14"></u-text>
  84. </view>
  85. </view>
  86. </view>
  87. <!-- end -->
  88. </template>
  89. </oa-scroll>
  90. <oa-tabbar :tabbarValue="0" :tabbarList="proxy.$constData.projectTabbar" :isSwitchTab="false"></oa-tabbar>
  91. <u-popup :show="popup.show" mode="bottom" bgColor="#fff" :round="10" @close="popup.show = false">
  92. <view
  93. :style="{
  94. borderTopLeftRadius: '10px',
  95. borderTopRightRadius: '10px',
  96. overflow: 'hidden',
  97. }"
  98. >
  99. <u-button
  100. v-if="state.tabsCurrent == 1"
  101. class="custom-style"
  102. type="info"
  103. size="normal"
  104. text="编辑"
  105. :customStyle="{
  106. height: '50px',
  107. color: '#3c9cff',
  108. borderWidth: 0,
  109. borderRadius: 0,
  110. borderBottomWidth: '1px',
  111. }"
  112. @click="handleSubmit('update', eventList)"
  113. ></u-button>
  114. <u-button
  115. class="custom-style"
  116. type="info"
  117. size="normal"
  118. text="一键复制"
  119. :customStyle="{
  120. height: '50px',
  121. color: '#3c9cff',
  122. borderWidth: 0,
  123. borderRadius: 0,
  124. borderBottomWidth: '1px',
  125. }"
  126. @click="handleSubmit('copy', eventList)"
  127. ></u-button>
  128. <u-button
  129. v-if="state.tabsCurrent == 1"
  130. class="custom-style"
  131. type="info"
  132. size="normal"
  133. text="删除"
  134. :customStyle="{
  135. height: '50px',
  136. color: '#f56c6c',
  137. borderWidth: 0,
  138. borderRadius: 0,
  139. borderBottomWidth: '3px',
  140. }"
  141. @click="handleModal('delete', `确认删除“ ${eventList.createBy} ” 的“ ${eventList.reportDate} ” 的日报?`)"
  142. ></u-button>
  143. <u-button
  144. class="custom-style"
  145. type="info"
  146. size="normal"
  147. :text="`查看${eventList.createBy}的所有日报`"
  148. :customStyle="{
  149. height: '50px',
  150. color: '#3c9cff',
  151. borderWidth: 0,
  152. borderRadius: 0,
  153. borderBottomWidth: '1px',
  154. }"
  155. @click="handleSubmit('update', eventList)"
  156. ></u-button>
  157. <u-button
  158. class="custom-style"
  159. type="info"
  160. size="normal"
  161. text="取消"
  162. :customStyle="{
  163. height: '50px',
  164. color: '#3c9cff',
  165. border: 'none',
  166. borderRadius: 0,
  167. }"
  168. @click="popup.show = false"
  169. ></u-button>
  170. </view>
  171. </u-popup>
  172. <uni-popup ref="alertDialog" type="dialog">
  173. <uni-popup-dialog :type="state.tip.type" cancelText="取消" confirmText="确定" title="操作提醒" :content="state.tip.content" @confirm="dialogConfirm" @close="dialogClose"></uni-popup-dialog>
  174. </uni-popup>
  175. <u-modal
  176. :show="modal.show"
  177. title="操作提醒"
  178. :content="modal.content"
  179. :showCancelButton="true"
  180. :closeOnClickOverlay="true"
  181. @confirm="handleSubmit(state.modal.type, eventList)"
  182. @cancel="modal.show = false"
  183. @close="modal.show = false"
  184. ></u-modal>
  185. </template>
  186. <script setup>
  187. /*----------------------------------依赖引入-----------------------------------*/
  188. import { onLoad, onShow, onReady, onHide, onLaunch, onUnload, onNavigationBarButtonTap, onPageScroll } from "@dcloudio/uni-app";
  189. import { ref, reactive, computed, getCurrentInstance, toRefs, inject } from "vue";
  190. /*----------------------------------接口引入-----------------------------------*/
  191. import { projectApi } from "@/api/business/project.js";
  192. /*----------------------------------组件引入-----------------------------------*/
  193. /*----------------------------------store引入-----------------------------------*/
  194. import { useStores, commonStores } from "@/store/modules/index";
  195. /*----------------------------------公共方法引入-----------------------------------*/
  196. /*----------------------------------公共变量-----------------------------------*/
  197. const { proxy } = getCurrentInstance();
  198. const useStore = useStores();
  199. /*----------------------------------变量声明-----------------------------------*/
  200. const state = reactive({
  201. tabsList: [
  202. { name: "我收到的", value: 2 },
  203. { name: "我发出的", value: 3 },
  204. { name: "我负责的", value: 1 },
  205. ],
  206. tabsCurrent: 0,
  207. loading: false,
  208. reportListNewData: [],
  209. pageSize: 20,
  210. current: 1,
  211. total: 0,
  212. modalShow: false,
  213. modalType: "",
  214. modalEvent: {},
  215. popup: {
  216. show: false, //弹窗显示
  217. content: "", //提示信息
  218. },
  219. modal: {
  220. type: "", //操作类型
  221. show: false, //弹窗显示
  222. content: "", //提示信息
  223. },
  224. eventList: {}, //数据存储
  225. reportData: {},
  226. tip: {
  227. type: undefined, //弹框类型
  228. content: "", //提示信息
  229. data: {}, //带入数据
  230. operation: undefined, //操作类型
  231. },
  232. timedList: [],
  233. tree: [],
  234. });
  235. const { tabsList, tabsCurrent, reportListNewData, pageSize, current, total, popup, eventList, modal, timedList, tree } = toRefs(state);
  236. /**
  237. * 操作弹框提醒
  238. * @param type 弹框类型
  239. * @param content 提示内容
  240. * @param item 带入数据
  241. * @param operation 操作类型
  242. */
  243. function tips(type, content, item, operation) {
  244. state.tip.type = type;
  245. state.tip.content = content;
  246. state.tip.data = item;
  247. state.tip.operation = operation;
  248. proxy.$refs.alertDialog.open();
  249. }
  250. /**弹框确定操作 */
  251. function dialogConfirm() {
  252. if (state.tip.operation == "delete") {
  253. handleSubmit("delete", state.tip.data);
  254. }
  255. if (state.tip.operation == "exit") {
  256. handleSubmit("exit", state.tip.data);
  257. }
  258. if (state.tip.operation == "copy") {
  259. handleSubmit("copy", state.tip.data);
  260. }
  261. }
  262. /**
  263. * @初始化
  264. */
  265. function init() {
  266. let arrayList = [];
  267. projectApi()
  268. .ReportRecord({
  269. // startDate: "2024-07-10",
  270. // endDate: "2024-07-10",
  271. pageNum: state.current,
  272. pageSize: state.pageSize,
  273. projectAscription: state.tabsList[state.tabsCurrent].value,
  274. })
  275. .then((requset) => {
  276. requset.data.records.forEach((el) => {
  277. el.contentText = "";
  278. el.workContents.forEach((cl) => {
  279. el.contentText += `${cl.projectName} ${cl.workTime}h \n ${cl.workContent} \n`;
  280. });
  281. });
  282. state.total = requset.data.total;
  283. state.loading = false;
  284. var reportListData = requset.data.records;
  285. var groupBySubmitTime = reportListData.reduce((acc, current) => {
  286. const existIndex = acc.findIndex((item) => item.submitDate.slice(0, 10) === current.submitDate.slice(0, 10));
  287. if (existIndex === -1) {
  288. acc.push({ submitDate: current.submitDate, items: [current] });
  289. } else {
  290. acc[existIndex].items.push(current);
  291. }
  292. return acc;
  293. }, []);
  294. groupBySubmitTime.forEach(function (item, index) {
  295. item.items.unshift({ submitDate: item.submitDate });
  296. item.items.forEach(function (aa) {
  297. arrayList.push(aa);
  298. });
  299. });
  300. state.reportListNewData = arrayList;
  301. })
  302. .catch((err) => {
  303. state.loading = false;
  304. });
  305. projectApi()
  306. .TimedReports({})
  307. .then((requset) => {
  308. state.timedList = requset.data;
  309. })
  310. .catch((err) => {});
  311. }
  312. /**
  313. * @跳转详情
  314. */
  315. function goContentDetails(e) {
  316. if (state.tabsCurrent == 0 && e.readFlag != 1) {
  317. projectApi()
  318. .ReportRecordReadFlag({ reportId: e.id })
  319. .then((res) => {
  320. if (res.status == "SUCCESS") {
  321. proxy.$tab.navigateTo(`/pages/business/common/projectMange/record/details?reportId=${e.id}`);
  322. } else {
  323. proxy.$modal.msgError("读取异常");
  324. }
  325. });
  326. } else {
  327. proxy.$tab.navigateTo(`/pages/business/common/projectMange/record/details?reportId=${e.id}`);
  328. }
  329. }
  330. /**
  331. * @scrollView加载数据
  332. */
  333. function load() {
  334. state.pageSize += 10;
  335. init();
  336. }
  337. /**
  338. * @scrollView刷新数据
  339. */
  340. function refresh() {
  341. state.pageSize = 20;
  342. init();
  343. }
  344. /**
  345. * @tabs点击事件
  346. */
  347. function tabsClick(e) {
  348. state.tabsCurrent = e.index;
  349. init();
  350. }
  351. // 定时日志页面
  352. function goTimingPage() {
  353. proxy.$tab.navigateTo(`/pages/business/common/projectMange/record/timingLog`);
  354. // proxy.$tab.navigateTo(`/pages/business/common/projectMange/report/timingLog`);
  355. }
  356. /** 更多按钮点击事件 */
  357. function moreClick(event) {
  358. state.popup.show = true;
  359. state.eventList = event;
  360. }
  361. function handleModal(type, content) {
  362. state.modal.show = true;
  363. state.modal.type = type;
  364. state.modal.content = content;
  365. }
  366. /** 编辑、删除日报*/
  367. function handleSubmit(type, item) {
  368. if (type === "update") {
  369. proxy.$tab.navigateTo(`/pages/business/common/projectMange/write/insert?templateId=1&id=${item.id}`);
  370. state.modal.show = false;
  371. } else if (type === "copy") {
  372. var workLongString = "";
  373. item.workContents.forEach((item) => {
  374. workLongString += item.projectName + ":" + item.workTime + "h\n" + item.workContent + "\n";
  375. });
  376. item.ccTo1 = item.ccTo.split(",").map(function (value, index) {
  377. return Number(value);
  378. });
  379. /** 查询树结构用户列表 回显抄送人*/
  380. item.tomorrowPlan = item.tomorrowPlan ? item.tomorrowPlan : "-";
  381. item.coordinateWork = item.coordinateWork ? item.coordinateWork : "-";
  382. // 触发方法
  383. proxy.$common.uniCopy({
  384. content: workLongString + "\n" + "明日计划:\n" + item.tomorrowPlan + "\n" + "工作协调:\n" + item.coordinateWork,
  385. success: (res) => {
  386. uni.showToast({
  387. title: res,
  388. icon: "none",
  389. });
  390. },
  391. error: (e) => {
  392. uni.showToast({
  393. title: e,
  394. icon: "none",
  395. duration: 3000,
  396. });
  397. },
  398. });
  399. state.popup.show = false;
  400. } else if (type === "delete") {
  401. projectApi()
  402. .ReportDelete(item.id)
  403. .then(() => {
  404. proxy.$modal.msg("日报删除成功!");
  405. state.modal.show = false;
  406. state.popup.show = false;
  407. init();
  408. })
  409. .catch((errors) => {
  410. proxy.$modal.msg(errors);
  411. });
  412. }
  413. state.modalShow = false;
  414. }
  415. onReady(() => {});
  416. onShow(() => {
  417. state.popup.show = false;
  418. });
  419. onLoad((options) => {
  420. init();
  421. uni.$on("projectMange_record", function (value) {
  422. init();
  423. });
  424. });
  425. onUnload(() => {
  426. uni.$off("projectMange_record"); //将值删除监听器
  427. });
  428. </script>
  429. <style lang="scss" scoped>
  430. .content-area {
  431. &-time {
  432. padding: 10px;
  433. text-align: left;
  434. color: #000000;
  435. font-weight: 600;
  436. }
  437. &-center {
  438. margin: 0;
  439. padding: 15px;
  440. overflow: hidden;
  441. border-bottom: 1px solid #eaeef1;
  442. &:last-child {
  443. border-bottom: 0px solid #eaeef1;
  444. }
  445. &-avatar {
  446. margin: auto 0;
  447. }
  448. &-avatarImg {
  449. width: 35px;
  450. height: 35px;
  451. border-radius: 4px;
  452. }
  453. &-title {
  454. margin: 0 0 15px 0;
  455. font-weight: 600;
  456. color: #000000;
  457. }
  458. }
  459. }
  460. .pp {
  461. text-align: left;
  462. }
  463. </style>
  464. <style>
  465. .pp .u-modal__content {
  466. justify-content: left !important;
  467. }
  468. </style>