detail.vue 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452
  1. <template>
  2. <u-navbar :autoBack="false" :placeholder="true" :safeAreaInsetTop="true" :bgColor="proxy.$settingStore.themeColor.color">
  3. <template #left>
  4. <u-icon name="arrow-left" size="20" color="#fff" @click="returnTo('business/oa/toDo/index')"></u-icon>
  5. </template>
  6. <template #center>
  7. <text class="grid-area_center_item_title" style="color: #fff;">详情</text>
  8. </template>
  9. </u-navbar>
  10. <view class="menu-list m0" style="overflow-y:scroll">
  11. <view class="list-cell" style="color: #666666; line-height: 25px; width: auto; ">
  12. <view class="content-area-top menu-item">
  13. <view class="content-area-top-name" >{{ state?.detail?.createBy + '提交的' + state?.detail?.formName }} </view>
  14. </view>
  15. <view class="content-area-row_wrap menu-item">
  16. <img class="typeIcon" src="@/static/images/oa/approval.png" v-if="state.approvalStatus === '1' "/>
  17. <img class="typeIcon" src="@/static/images/oa/pass.png" v-if="state.approvalStatus === '2' "/>
  18. <img class="typeIcon" src="@/static/images/oa/refuse.png" v-if="state.approvalStatus === '3' "/>
  19. <view class="content-area-row_wrap-view gray"> 上海永天科技股份有限公司 </view>
  20. <view class="content-area-row_wrap-view gray"> 单据编号:{{ state.detail.docNo }} </view>
  21. <view class="content-area-row_wrap-view" style="display: block">
  22. <span :style="{ color: proxy.$common.mapping('elTagClass', 'value', state.approvalStatus, form_status) }" >{{ proxy.$common.mapping("label", "value", state.approvalStatus, form_status) }} </span>
  23. </view>
  24. <view class="content-area-row_wrap-view gray label">所在部门</view>
  25. <view class="content-area-row_wrap-view gray black">{{ state.detail.deptName }}</view>
  26. <view class="content-area-row_wrap-view gray label" v-if="state.detail.duration">{{ state.detail.docNo.split("-")[0] == "JBD" ? "加班" : state.detail.docNo.split("-")[0] == "QJD" ? "请假" : "" }}时长</view>
  27. <view class="content-area-row_wrap-view gray black" v-if="state.detail.duration">{{ state.detail.duration }}</view>
  28. <view class="content-area-row_wrap-view gray label" v-if="state.detail.reason">{{ state.detail.docNo.split("-")[0] == "JBD" ? "加班" : state.detail.docNo.split("-")[0] == "QJD" ? "请假" : "" }}原因</view>
  29. <view class="content-area-row_wrap-view gray black" v-if="state.detail.reason">{{ state.detail.reason }}</view>
  30. </view>
  31. </view>
  32. <view class="list-cell" style="color: #666666; line-height: 25px; width: auto;margin:10px 0 60px 0;">
  33. <view class="content-area-top">
  34. <view style="width:100%;">流程</view>
  35. <view class="stepBar">
  36. <view class="item">
  37. <view class="left">
  38. <image class="icon" src="@/static/images/oa/user.png"></image>
  39. <image class="status" src="@/static/images/oa/pass2.png"></image>
  40. <div>
  41. <view class="title">
  42. 发起申请
  43. </view>
  44. <view class="name">
  45. {{ state.detail.createBy }}
  46. </view>
  47. </div>
  48. </view>
  49. <view class="right">
  50. {{ state.detail.createTime }}
  51. </view>
  52. <view class="line sucess">
  53. </view>
  54. </view>
  55. <view class="item" v-for="(item, index) in nodeList" :key="index">
  56. <view class="left">
  57. <image class="icon" src="@/static/images/oa/user.png" v-if="item.nodeType == 1"></image>
  58. <image class="icon" src="@/static/images/oa/Ccto.png" v-if="item.nodeType == 2"></image>
  59. <div>
  60. <view class="title">
  61. {{ item.nodeType == 1 ? '审批人' : item.nodeType == 2 ? `抄送${item.appointApprover.indexOf(",") && item.appointApprover.split(",").length>1 ? `(${item.appointApprover.split(",").length}人)` : "人"}` : "" }}
  62. </view>
  63. <view class="name" v-if="item.nodeType == 1">
  64. {{ proxy.$common.mapping("nickName", "userId", item.appointApprover, state.userData) }}
  65. </view>
  66. <view class="name" v-if="item.nodeType == 2 && item.appointApprover.indexOf(',') && item.appointApprover.split(',').length == 1">
  67. {{ proxy.$common.mapping("nickName", "userId", item.appointApprover, state.userData) }}
  68. </view>
  69. <view class="name" v-if="item.nodeType == 2 && item.appointApprover.indexOf(',') && item.appointApprover.split(',').length > 1">
  70. 抄送人
  71. </view>
  72. </div>
  73. </view>
  74. <view class="right">
  75. <!-- 2021-08-01 10:30:00 -->
  76. </view>
  77. <view class="center" v-if="item.nodeType == 2 && item.appointApprover.indexOf(',') && item.appointApprover.split(',').length > 1">
  78. <view class="content-area-header mb10 text-center" style="display: inline-block" v-for="(approver, index2) in item.appointApprover.split(',')" :key="index2">
  79. <img v-if='proxy.$common.mapping("avatar", "userId", approver, state.userData)' class="content-area-header-avatarImg mlr5" :src='proxy.$common.mapping("avatar", "userId", approver, state.userData)' style="display: block; width: 40px; height: 40px" />
  80. <u-avatar
  81. class="content-area-header-avatar mlr5"
  82. :text='proxy.$common.mapping("nickName", "userId", approver, state.userData).length > 2 ? proxy.$common.mapping("nickName", "userId", approver, state.userData).slice(1, 3) : proxy.$common.mapping("nickName", "userId", approver, state.userData)'
  83. shape="square"
  84. size="40"
  85. fontSize="12"
  86. color="#ffffff"
  87. :bgColor="proxy.$settingStore.themeColor.color"
  88. ></u-avatar>
  89. <u-text :text='proxy.$common.mapping("nickName", "userId", approver, state.userData)' color="#000000" size="14" align="center"></u-text>
  90. </view>
  91. </view>
  92. <view class="line gray" v-if="item.nodeType == 1">
  93. </view>
  94. </view>
  95. </view>
  96. </view>
  97. </view>
  98. <view class="fixedBottom" v-if="state.approvalStatus == 2 || state.approvalStatus == 3">
  99. <img src="@/static/images/oa/loading.png" alt="">
  100. <span>再次提交</span>
  101. </view>
  102. <view class="approval" v-if="state.approvalStatus == 1">
  103. <button class="refuse" @click="handle('refuse',2)">拒绝</button>
  104. <button class="pass" @click="handle('pass',2)">同意</button>
  105. </view>
  106. </view>
  107. </template>
  108. <script setup>
  109. /*----------------------------------依赖引入-----------------------------------*/
  110. import { onLoad, onShow, onReady, onHide, onLaunch, onNavigationBarButtonTap, onPageScroll } from "@dcloudio/uni-app";
  111. import { ref, reactive, computed, getCurrentInstance, toRefs, inject, watchEffect } from "vue";
  112. /*----------------------------------接口引入-----------------------------------*/
  113. import { getToDoPageList, approve } from "@/api/oa/todo/index.js";
  114. import { getFormName, getDocumentDetails,getOaFormDefinition } from "@/api/oa/approval/index.js";
  115. import { listDept, UserList } from "@/api/system/user";
  116. /*----------------------------------组件引入-----------------------------------*/
  117. /*----------------------------------store引入-----------------------------------*/
  118. /*----------------------------------公共方法引入-----------------------------------*/
  119. /*----------------------------------公共变量-----------------------------------*/
  120. const { proxy } = getCurrentInstance();
  121. const { form_status, workflow_form_subset } = proxy.useDict("form_status","workflow_form_subset");
  122. /*----------------------------------变量声明-----------------------------------*/
  123. const state = reactive({
  124. loading: false,
  125. queryParams:{
  126. docNo: undefined,
  127. pageNum:1,
  128. pageSize:5,
  129. },
  130. detail:{},//详情数据
  131. userData:[],//用户列表
  132. nodeList:[],//节点列表
  133. approvalStatus:"",
  134. id:undefined,
  135. });
  136. const { queryParams, detail,formList, userData, nodeList, approvalStatus, id } = toRefs(state);
  137. /**
  138. * @页面初始化
  139. */
  140. function init() {
  141. getDetail();
  142. }
  143. /**
  144. * 表单审核
  145. * @param type
  146. * @param status
  147. */
  148. function handle(type) {
  149. if(type == "pass" || type == "refuse"){
  150. approve({id:state.detail.id,approvalStatus:type == "pass"? 2:3}).then(() => {
  151. proxy.$modal.msgSuccess("操作成功");
  152. uni.redirectTo({
  153. url: `/pages/business/oa/toDo/index`
  154. })
  155. })
  156. }
  157. }
  158. /**
  159. * @列表查询
  160. * @api接口查询
  161. */
  162. function getDetail(type) {
  163. state.loading = true;
  164. // getFormName().then((res1) => {
  165. // state.formList = res1.data;
  166. getDocumentDetails(state.queryParams)
  167. .then((requset) => {
  168. state.detail = requset.data;
  169. state.detail.id = state.id;
  170. listDept({id:state.detail.deptId}).then((res2) => {
  171. state.detail.deptName = res2.data[0].deptName
  172. })
  173. UserList({ pageNum: "1", pageSize: "10000" }).then((res3) => {
  174. state.userData = res3.data.rows;
  175. console.log(state.userData.length)
  176. getOaFormDefinition( {formSign:state.queryParams?.docNo?.split("-")[0]}).then((res4)=>{
  177. if(res4.data.length){
  178. var info = res4.data[0]
  179. state.nodeList = JSON.parse(info.flowInfo).node
  180. state.detail.formName = info.formName
  181. }
  182. })
  183. })
  184. state.loading = false;
  185. })
  186. .catch((err) => {
  187. state.loading = false;
  188. });
  189. // })
  190. }
  191. /**
  192. * 返回上级页面
  193. * @param defaultPage 默认页面
  194. */
  195. function returnTo(defaultPage) {
  196. if(getCurrentPages().length > 1){
  197. uni.navigateBack()
  198. }else{
  199. uni.redirectTo({
  200. url: `/pages/${defaultPage}`
  201. })
  202. }
  203. }
  204. onShow((options) => {
  205. });
  206. onLoad((options) => {
  207. if(options?.docNo){
  208. state.queryParams.docNo = options.docNo;
  209. }
  210. if(options?.approvalStatus){
  211. state.approvalStatus = options.approvalStatus;
  212. }
  213. if(options?.id){
  214. state.id = options.id;
  215. }
  216. init();
  217. //调用系统主题颜色
  218. proxy.$settingStore.systemThemeColor([1]);
  219. });
  220. </script>
  221. <style lang="scss" scoped>
  222. :deep(.u-modal__content) {
  223. font-size: 14px;
  224. justify-content: left;
  225. }
  226. :deep(.list-container .content-area-top-name) {
  227. font-size: 16px !important;
  228. }
  229. </style>
  230. <style lang="scss" scoped>
  231. .content-area {
  232. &-top {
  233. font-size: 16px;
  234. font-weight: 600;
  235. color: #000000;
  236. width: 100%;
  237. &-name {
  238. font-size: 13px;
  239. width: 65%;
  240. text-align: left;
  241. color: #000;
  242. white-space: nowrap; /* 确保文本在一行内显示 */
  243. overflow: hidden; /* 隐藏超出容器的内容 */
  244. text-overflow: ellipsis; /* 使用省略号表示被截断的文本 */
  245. }
  246. &-time {
  247. font-size: 12px;
  248. text-align: right;
  249. color: #999;
  250. float:right;
  251. width:35%;
  252. }
  253. &-status {
  254. max-width: 30%;
  255. margin: auto 0 auto auto;
  256. font-size: 12px;
  257. color: #ffffff;
  258. padding: 0 5px;
  259. border-radius: 20px;
  260. line-height: 20px;
  261. }
  262. &-icon {
  263. max-width: 30%;
  264. margin: auto 0 auto auto;
  265. }
  266. }
  267. &-row_wrap {
  268. font-size: 13px;
  269. flex-flow: row wrap;
  270. &-view {
  271. display: flex;
  272. min-width: 100%;
  273. > .iconfont {
  274. font-size: 14px;
  275. color: #909399;
  276. margin-left: 5px;
  277. }
  278. &-status {
  279. margin: auto 0 auto 0;
  280. font-size: 12px;
  281. font-weight: 600;
  282. color: #ffffff;
  283. padding: 0 5px;
  284. border-radius: 20px;
  285. line-height: 20px;
  286. }
  287. }
  288. }
  289. }
  290. .gray{
  291. color:#999;
  292. }
  293. .label{
  294. margin-top:10px;
  295. }
  296. .black{
  297. color:#000;
  298. margin-top:0px;
  299. }
  300. .typeIcon{
  301. width:55px;
  302. position: absolute;
  303. top:25px;
  304. right:10px;
  305. }
  306. .fixedBottom{
  307. width:100%;
  308. height:36px;
  309. line-height: 36px;
  310. position: fixed;
  311. bottom: 0;
  312. left:0;
  313. background-color: #fff;
  314. text-align: center;
  315. color:#000;
  316. img{
  317. vertical-align: middle;
  318. margin-right:10px;
  319. margin-top:-2px;
  320. }
  321. span{
  322. vertical-align: middle;
  323. }
  324. }
  325. .stepBar{
  326. margin-top:-20px;
  327. font-weight: 400;
  328. .item{
  329. position: relative;
  330. margin-top:10%;
  331. .left,.right{
  332. display: inline-block;
  333. width:50%;
  334. }
  335. .left{
  336. position: relative;
  337. .icon{
  338. width:40px;
  339. height:40px;
  340. vertical-align: middle;
  341. float: left;
  342. }
  343. .status{
  344. width:16px;
  345. height:16px;
  346. position: absolute;
  347. left:28px;
  348. bottom:-2px;
  349. }
  350. >div{
  351. width:80%;
  352. margin-left:50px;
  353. vertical-align: middle;
  354. line-height: 20px;
  355. .title{
  356. font-size: 14px;
  357. }
  358. .name{
  359. font-size: 12px;
  360. color: #999;
  361. }
  362. }
  363. }
  364. .right{
  365. vertical-align: top;
  366. font-size: 12px;
  367. text-align: right;
  368. color:#909399;
  369. }
  370. .line{
  371. height:40px;
  372. width:1px;
  373. position: absolute;
  374. top:40px;
  375. left:18px;
  376. }
  377. .line.sucess{
  378. background: #67c23a;
  379. }
  380. .line.gray{
  381. background: #999;
  382. }
  383. .center{
  384. width:100%;
  385. margin:10px 0 0 40px;
  386. padding-right:10px;
  387. }
  388. }
  389. }
  390. .content-area {
  391. margin: 0;
  392. padding: 15px 20px;
  393. overflow: hidden;
  394. &-header {
  395. &-avatar {
  396. margin: auto 0;
  397. }
  398. &-avatarImg {
  399. width: 35px;
  400. height: 35px;
  401. border-radius: 4px;
  402. }
  403. &-title {
  404. margin: 0 0 15px 0;
  405. font-weight: 600;
  406. color: #000000;
  407. }
  408. }
  409. &-center {
  410. line-height: 25px;
  411. &-top {
  412. color: #000000;
  413. font-weight: 600;
  414. }
  415. }
  416. }
  417. .approval{
  418. position: fixed;
  419. width:100%;
  420. bottom: 10px;
  421. left:0;
  422. button{
  423. width:calc(50% - 15px);
  424. height:50px;
  425. margin-left:10px;
  426. display: inline-block;
  427. border-radius: 16px !important;
  428. }
  429. button.pass{
  430. background: #2A98FF;
  431. color:#fff;
  432. }
  433. button.refuse{
  434. border:1px solid #CFCFCF;
  435. }
  436. }
  437. </style>