rapidRep.vue 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591
  1. <template>
  2. <view class="handler">
  3. <view class="body view-body">
  4. <view class="form_item column">
  5. <view class="title"><text class="required newicon newicon-bitian"></text>故障描述:</view>
  6. <uni-easyinput class="value" type="textarea" v-model="dataInfo.deferralRemark" placeholder="请输入故障描述" :class="{formRed: isSubmit && !dataInfo.deferralRemark.trim()}" />
  7. </view>
  8. <view class="candidate">
  9. <view class="candidate-item" v-for="item in candidateData" :key="item" @click="itemCandidate(item)">{{item.name}}</view>
  10. </view>
  11. <view class="form_item" v-if="cmdbRepair.valueconfig==1">
  12. <view class="title select">关联资产:</view>
  13. <input class="item-input" focus placeholder="请扫描资产卡二维码" v-model="dataInfo.property"/>
  14. <text class="newicon newicon-saoma icon" @click="scanCodes"></text>
  15. </view>
  16. <view class="form_item">
  17. <view class="title">照片录像:</view>
  18. <view class="value">
  19. <uni-file-picker ref="handlerImgRef" v-model="dataInfo.handlerImgList"
  20. limit="4" @success="handlerImgSuccess"
  21. @fail="handlerImgFail" @select="handlerImgSelect" @delete="handlerImgDelete">
  22. </uni-file-picker>
  23. <view class="imgTips">(支持JPG/PNG格式图片,单张大小3M以内,录像支持30秒)</view>
  24. </view>
  25. </view>
  26. <view class="form_item">
  27. <view class="title">录音:</view>
  28. <view class="chunk" v-if="!dataInfo.recBlob" @click="examineRecord" @mouseup="recStop">按住录音</view>
  29. <view v-if="dataInfo.recBlob">
  30. <!-- <audio src="dataInfo.recBlob" controls></audio> -->
  31. <icon type="clear" size="26" @click="clearRec"/>
  32. </view>
  33. </view>
  34. </view>
  35. <view class="foot_common_btns">
  36. <button @click="goBackOrToList" type="default" class="cancelButton btn">返回</button>
  37. <button @click="submit" type="default" class="primaryButton btn">下一步</button>
  38. </view>
  39. </view>
  40. </template>
  41. <script setup>
  42. import { SM } from "@/http/http.js"
  43. import { ref, reactive, computed } from 'vue'
  44. import NumberModal from '@/components/NumberModal.vue';
  45. import { onLoad } from '@dcloudio/uni-app'
  46. import { generateNumberArray } from '@/utils/index.js'
  47. import { api_group, api_incidentDetail, api_getSolution, api_user, api_incidentTask, api_branch, api_dutyDepartment, api_getDictionary, api_querySummaryDoc, api_addSummaryDoc } from "@/http/api.js"
  48. import { defaultColor } from '@/static/js/theme.js'
  49. import { useSetTitle } from '@/share/useSetTitle.js'
  50. import { useMakePhoneCall } from '@/share/useMakePhoneCall.js'
  51. import { useUploadFile } from '@/share/useUploadFile.js'
  52. import { useGoBack } from '@/share/useGoBack.js'
  53. import { useLoginUserStore } from '@/stores/loginUser'
  54. import { useHandlerStore } from '@/stores/handler'
  55. import Recorder from 'recorder-core';
  56. import 'recorder-core/src/engine/mp3';
  57. import 'recorder-core/src/engine/mp3-engine';
  58. import 'recorder-core/src/engine/wav';
  59. import 'recorder-core/src/extensions/waveview';
  60. useSetTitle();
  61. const loginUserStore = useLoginUserStore();
  62. const handlerStore = useHandlerStore();
  63. const { makePhoneCall } = useMakePhoneCall();
  64. const { uploadFile } = useUploadFile();
  65. const { goBack } = useGoBack();
  66. // 主题颜色
  67. const primaryColor = ref(defaultColor)
  68. // 数据
  69. const dataInfo = reactive({
  70. tabs: [
  71. // {id: 5, name: '故障处理', value: 'doing', num: ''},
  72. // {id: 6, name: '延期处理', value: 'overtime', num: ''},
  73. ],
  74. tabActiveValue: 0,//当前选择的tab
  75. incidentId: undefined,//事件ID
  76. incidentData: {},//事件对象
  77. deferralRemark: '',//故障描述
  78. property:'', //资产
  79. handlerImgList: [],//处理图片列表
  80. recBlob:'' //录音
  81. })
  82. // 故障处理用是否提供备用机
  83. const newProvideBackupMachine = ref(0)
  84. // 知识库id
  85. const solutionId = ref(null)
  86. // 是否提交
  87. const isSubmit = ref(false)
  88. // 处理图片
  89. const handlerImgRef = ref(null)
  90. const candidateData = ref([
  91. {name:'马桶肃杀'},
  92. {name:'马桶肃杀'},
  93. {name:'马桶肃杀'},
  94. {name:'马桶肃杀'},
  95. {name:'马桶肃杀'},
  96. {name:'马桶肃杀'},
  97. {name:'马桶肃杀'},
  98. {name:'马桶肃杀'},
  99. {name:'马桶肃杀'},
  100. ])
  101. const cmdbRepair = ref(null)
  102. let rec = null;
  103. let wave = null;
  104. const recwave = ref(null);
  105. // 上一步或者返回列表
  106. function goBackOrToList(){
  107. uni.setStorageSync('repairData','')
  108. goBack();
  109. }
  110. function itemCandidate(item){
  111. dataInfo.deferralRemark = item.name
  112. }
  113. // 获取录音权限
  114. function recOpen() {
  115. rec = Recorder({
  116. type: 'wav', //录音格式,可以换成wav等其他格式
  117. sampleRate: 16000, //录音的采样率,越大细节越丰富越细腻
  118. bitRate: 16, //录音的比特率,越大音质越好
  119. onProcess: (buffers, powerLevel, bufferDuration, bufferSampleRate) => {
  120. if (wave) {
  121. wave.input(buffers[buffers.length - 1], powerLevel, bufferSampleRate);
  122. }
  123. },
  124. });
  125. if (!rec) {
  126. alert('当前浏览器不支持录音功能!');
  127. return;
  128. }
  129. //打开录音,获得权限
  130. rec.open(
  131. () => {
  132. console.log('录音已打开');
  133. startRecord()
  134. if (recwave.value) {
  135. //创建音频可视化图形绘制对象
  136. wave = Recorder.WaveView({ elem: recwave.value });
  137. }
  138. },
  139. (msg, isUserNotAllow) => {
  140. //用户拒绝了录音权限,或者浏览器不支持录音
  141. console.log((isUserNotAllow ? 'UserNotAllow,' : '') + '无法录音:' + msg);
  142. },
  143. );
  144. // uni.getSetting({
  145. // success(res) {
  146. // // 判断是否开启了录音权限
  147. // if (res.authSetting['scope.record']) {
  148. // startRecord()
  149. // } else {
  150. // uni.showToast({
  151. // icon: 'none',
  152. // title: '录音权限未开启,无法录音',
  153. // mask: true,
  154. // });
  155. // }
  156. // }
  157. // })
  158. }
  159. // 检查录音
  160. function examineRecord() {
  161. recOpen()
  162. }
  163. function startRecord(){
  164. if (!rec) {
  165. console.error('未打开录音');
  166. return;
  167. }
  168. rec.start();
  169. }
  170. // 结束录音
  171. function recStop() {
  172. if (!rec) {
  173. console.error('未打开录音');
  174. return;
  175. }
  176. rec.stop(
  177. (blob, duration) => {
  178. //blob就是我们要的录音文件对象,可以上传,或者本地播放
  179. dataInfo.recBlob = blob;
  180. //简单利用URL生成本地文件地址,此地址只能本地使用,比如赋值给audio.src进行播放,赋值给a.href然后a.click()进行下载(a需提供download="xxx.mp3"属性)
  181. const localUrl = (window.URL || window.webkitURL).createObjectURL(blob);
  182. console.log('录音成功', blob, localUrl, '时长:' + duration + 'ms');
  183. // upload(blob); //把blob文件上传到服务器
  184. rec.close(); //关闭录音,释放录音资源,当然可以不释放,后面可以连续调用start
  185. rec = null;
  186. },
  187. (err) => {
  188. console.error('结束录音出错:' + err);
  189. rec.close(); //关闭录音,释放录音资源,当然可以不释放,后面可以连续调用start
  190. rec = null;
  191. },
  192. );
  193. }
  194. // 删除录音
  195. function clearRec(){
  196. dataInfo.recBlob = null
  197. }
  198. // 扫码资产码
  199. function scanCodes(){
  200. SM().then((res) => {
  201. uni.showLoading({
  202. title: "加载中",
  203. mask: true,
  204. });
  205. uni.hideLoading();
  206. })
  207. }
  208. // 上传处理图片成功
  209. function handlerImgSuccess(e){
  210. dataInfo.handlerImgList.forEach(v => {
  211. v.url = v.path;
  212. })
  213. console.log(dataInfo.handlerImgList);
  214. let handlerOrder$ = handlerOrder();
  215. let requestList = [handlerOrder$];
  216. dataInfo.handlerImgList.forEach(v => {
  217. let handlerOrderImg$ = handlerOrderImg(v);
  218. requestList.push(handlerOrderImg$);
  219. })
  220. Promise.all(requestList).then(resList => {
  221. uni.hideLoading();
  222. console.log(resList);
  223. if(resList[0].state == 200){
  224. uni.showToast({
  225. icon: 'none',
  226. title: '处理成功',
  227. mask: true,
  228. });
  229. setTimeout(() => {
  230. uni.reLaunch({
  231. url: '/pages/incidentList/incidentList',
  232. })
  233. }, 1500)
  234. }else{
  235. uni.showToast({
  236. icon: 'none',
  237. title: resList[0].msg || '请求数据失败!'
  238. });
  239. }
  240. })
  241. }
  242. // 上传处理图片失败
  243. function handlerImgFail(e){
  244. dataInfo.handlerImgList.forEach(v => {
  245. v.url = v.path;
  246. })
  247. console.log(dataInfo.handlerImgList);
  248. }
  249. // 选择上传图片
  250. function handlerImgSelect(e){
  251. dataInfo.handlerImgList = dataInfo.handlerImgList.concat(e.tempFiles);
  252. console.log(dataInfo.handlerImgList);
  253. }
  254. // 删除上传图片
  255. function handlerImgDelete(e){
  256. dataInfo.handlerImgList = dataInfo.handlerImgList.filter(v => e.tempFile.uuid != v.uuid);
  257. console.log(dataInfo.handlerImgList);
  258. }
  259. // 获取事件详情
  260. function getIncidentDetail(){
  261. if(uni.getStorageSync('repairData')){
  262. let data = JSON.parse(uni.getStorageSync('repairData'))
  263. if(data){
  264. dataInfo.deferralRemark = data.deferralRemark
  265. dataInfo.property = data.property
  266. dataInfo.handlerImgList = data.handlerImgList
  267. dataInfo.recBlob = data.recBlob
  268. }
  269. }
  270. }
  271. // 下一步
  272. function submit(){
  273. isSubmit.value = true;
  274. if(dataInfo.deferralRemark==''){
  275. uni.showToast({
  276. icon: 'none',
  277. title: '请选择输入故障描述'
  278. });
  279. return;
  280. }
  281. uni.setStorageSync('repairData',JSON.stringify(dataInfo))
  282. uni.navigateTo({
  283. url: `/pages/repair/rapidRepNext`,
  284. });
  285. }
  286. // 处理提交事件
  287. function handlerOrder(){
  288. dataInfo.incidentData.returnBackupMachine = dataInfo.returnBackupMachine
  289. let postData = {
  290. incident: dataInfo.incidentData,
  291. solutionId:solutionId.value
  292. }
  293. postData.incident.handleDescription = dataInfo.handleDescription;
  294. postData.incident.handleCategory = {id: dataInfo.handleCategory};
  295. postData.incident.closecode = {id: dataInfo.closecode};
  296. postData.incident.category = dataInfo.category;
  297. postData.incident.synergetic = dataInfo.synergetic;
  298. return api_incidentTask(dataInfo.tabActiveValue, postData);
  299. }
  300. // 处理图片
  301. function handlerOrderImg(imgObj){
  302. return uploadFile(imgObj, 'incident', dataInfo.incidentId)
  303. }
  304. onLoad((option) => {
  305. // let storeData = handlerStore.handler.data
  306. // if(storeData && storeData.type=='rep'){
  307. // solutionId.value = storeData.solutionId
  308. // dataInfo.isSummaryNext = storeData.isSummaryNext
  309. // dataInfo.incidentId = storeData.incidentId;
  310. // }else{
  311. // dataInfo.incidentId = option.incidentId;
  312. // dataInfo.isSummaryNext = option.isSummaryNext == 1;
  313. // }
  314. let data = JSON.parse(uni.getStorageSync('sysData'))
  315. cmdbRepair.value = data.find(i=>i.keyconfig=='cmdbRepair')
  316. getIncidentDetail();
  317. })
  318. </script>
  319. <style lang="scss" scoped>
  320. .handler{
  321. height: 100%;
  322. display: flex;
  323. flex-direction: column;
  324. justify-content: space-between;
  325. padding: 0 30rpx;
  326. .head{
  327. height: 88rpx;
  328. display: flex;
  329. position: fixed;
  330. z-index: 99;
  331. width: 100%;
  332. background-color: #fff;
  333. font-size: 30rpx;
  334. .tab{
  335. flex: 1;
  336. display: flex;
  337. justify-content: center;
  338. align-items: center;
  339. border-bottom: 4rpx solid transparent;
  340. &.active{
  341. color: $uni-primary;
  342. border-color: $uni-primary;
  343. }
  344. }
  345. }
  346. .body{
  347. box-sizing: border-box;
  348. flex: 1;
  349. min-height: 0;
  350. &.bg{
  351. background-color: #F7F7F7;
  352. }
  353. .summaryItem{
  354. &:first-of-type{
  355. .summaryItem_head{
  356. border-bottom: 1rpx solid #DDDDDD;
  357. }
  358. }
  359. .summary_total{
  360. padding: 20rpx 0;
  361. display: flex;
  362. justify-content: center;
  363. align-items: center;
  364. }
  365. .summaryItem_head{
  366. padding: 24rpx;
  367. font-size: 26rpx;
  368. color: #3A3A3A;
  369. }
  370. .summaryItem_body{
  371. font-size: 30rpx;
  372. background-color: #fff;
  373. .summaryItem_bodyItem{
  374. padding: 24rpx;
  375. border-bottom: 1rpx solid #DDDDDD;
  376. .summaryItem_bodyItem_top{
  377. display: flex;
  378. justify-content: space-between;
  379. align-items: center;
  380. .value{
  381. padding-left: 48rpx;
  382. flex-shrink: 0;
  383. }
  384. }
  385. .summaryItem_bodyItem_bottom{
  386. margin-top: 24rpx;
  387. display: flex;
  388. justify-content: space-between;
  389. align-items: center;
  390. .name{
  391. text-align: right;
  392. flex: 1;
  393. }
  394. .value{
  395. width: 240rpx;
  396. text-align: right;
  397. padding-left: 48rpx;
  398. flex-shrink: 0;
  399. }
  400. }
  401. }
  402. }
  403. .summaryItem_foot{
  404. font-size: 30rpx;
  405. background-color: #fff;
  406. &.total{
  407. margin-top: 24rpx;
  408. }
  409. .summaryItem_foot_total{
  410. padding: 24rpx 0;
  411. display: flex;
  412. justify-content: center;
  413. align-items: center;
  414. }
  415. .summaryItem_foot_add{
  416. border-top: 1rpx solid #DDDDDD;
  417. padding: 24rpx 0;
  418. display: flex;
  419. justify-content: center;
  420. align-items: center;
  421. .newicon-icon-test{
  422. font-size: 30rpx;
  423. font-weight: bold;
  424. }
  425. }
  426. }
  427. }
  428. .form_item_column{
  429. padding-top: 24rpx;
  430. min-height: 86rpx;
  431. .form_item{
  432. padding-top: 0;
  433. min-height: auto;
  434. }
  435. }
  436. .candidate{
  437. display: flex;
  438. flex-wrap: wrap;
  439. .candidate-item{
  440. padding: 6rpx 15rpx;
  441. font-size: 26rpx;
  442. color: #949494;
  443. background: #E9E9E9;
  444. border-radius: 50rpx;
  445. margin-right: 20rpx;
  446. margin-top: 15rpx;
  447. }
  448. }
  449. .form_item{
  450. display: flex;
  451. align-items: center;
  452. padding-top: 24rpx;
  453. min-height: 86rpx;
  454. position: relative;
  455. .chunk{
  456. width: 100%;
  457. height: 70rpx;
  458. line-height: 70rpx;
  459. text-align: center;
  460. background: #F7F8FA;
  461. box-shadow: 0px 3px 6px 1px rgba(0,0,0,0.16);
  462. border-radius: 10rpx;
  463. }
  464. &.column{
  465. height: auto;
  466. flex-direction: column;
  467. align-items: flex-start;
  468. .import-rep{
  469. padding: 5rpx 10rpx;
  470. border-radius: 50rpx;
  471. background: #d1fcd5;
  472. color: #49b856;
  473. font-size: 24rpx;
  474. }
  475. .title{
  476. margin-right: 0;
  477. }
  478. .title-width{
  479. width: 100%;
  480. }
  481. .title-fl-sb{
  482. display: flex;
  483. justify-content: space-between;
  484. width: 100%;
  485. }
  486. .value{
  487. margin-top: 10rpx;
  488. // padding-left: 20rpx;
  489. box-sizing: border-box;
  490. }
  491. .tips{
  492. padding: 24rpx;
  493. text-align: center;
  494. font-size: 22rpx;
  495. color: #909399;
  496. width: 100%;
  497. box-sizing: border-box;
  498. }
  499. }
  500. .title{
  501. font-size: 26rpx;
  502. display: flex;
  503. align-items: center;
  504. margin-right: 12rpx;
  505. flex-shrink: 0;
  506. &.select{
  507. width: calc(5em + 20rpx);
  508. }
  509. }
  510. .value{
  511. width: 100%;
  512. &.category{
  513. width: 100%;
  514. display: flex;
  515. justify-content: space-between;
  516. align-items: center;
  517. .categoryName{
  518. font-size: 26rpx;
  519. color: #555;
  520. flex: 1;
  521. }
  522. .newicon-weibiaoti2010104{
  523. color: $uni-primary;
  524. margin-left: 24rpx;
  525. }
  526. }
  527. .imgTips{
  528. color: #909399;
  529. font-size: 22rpx;
  530. margin-top: 10rpx;
  531. }
  532. }
  533. .item-input{
  534. border: 1px solid #DBDBDB;
  535. height: 70rpx;
  536. line-height: 70rpx;
  537. padding: 0 10rpx;
  538. border-radius: 4rpx;
  539. width: 100%;
  540. font-size: 14px;
  541. }
  542. .icon{
  543. position: absolute;
  544. right: 20rpx;
  545. color: #49b856;
  546. }
  547. .synergeticNames{
  548. font-size: 26rpx;
  549. margin-right: 24rpx;
  550. }
  551. .synergeticAdd{
  552. flex-shrink: 0;
  553. }
  554. }
  555. }
  556. }
  557. </style>