createdBug.vue 31 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865
  1. <template>
  2. <div class="Parent">
  3. <el-dialog :visible.sync="modalShow" :title="titleName" class="public_task" width="70%" :close-on-click-modal="false" :destroy-on-close="true" :modal-append-to-body="false" @close="modalClose">
  4. <div class="blueStripe" />
  5. <el-form ref="formInline" label-position="left" :model="formInline" :rules="rules" label-width="90px">
  6. <div id="soll" style="height: 460px; overflow:scroll; overflow-x: hidden">
  7. <div>
  8. <el-row style="margin: 0 4%;">
  9. <el-col :span="24">
  10. <el-form-item label="标题" prop="bugName">
  11. <el-input v-model="formInline.bugName" placeholder="请输入" style="width:100%;" />
  12. </el-form-item>
  13. </el-col>
  14. </el-row>
  15. <div class="Layouts" style="width: 100%;">
  16. <div style="width:100%; margin: 0 4%;">
  17. <el-form-item label="所属任务" prop="taskId">
  18. <el-select v-model="formInline.taskId" filterable :remote="true" :remote-method="debounceQuery" placeholder="请选择" style="width:100%;" @click.native="bugListSelect">
  19. <el-option v-for="item in taskEnumList" :key="item.id" :label="item.name" :value="item.id">
  20. <div class="belong-task">
  21. <div class="task-id">{{ item.taskId }}</div>
  22. <div class="modules-name">
  23. <span class="name">{{ item.name }}</span>
  24. <span v-if="item.moduleInfoName" class="modules">{{ item.moduleInfoName }}</span>
  25. </div>
  26. </div>
  27. </el-option>
  28. </el-select>
  29. </el-form-item>
  30. <el-form-item label="优先级" prop="priorityLevel" style="white-space: nowrap;">
  31. <el-tooltip class="item" effect="dark" content="i. High(阻塞阻塞进程的bug、主流程的严重bug,如app启动失败、接单crash,当天必须立即修复) ii. Medium(功bug,新需求的功能性bug建议放在该等级,可协商修复期限) iv. Low(不影响功能使用的小问题,如界面显示有异常,文案、UI微调,可协商修复期限)" placement="bottom">
  32. <i style="color: red; margin-left: -31px;" class="el-icon-warning-outline" />
  33. </el-tooltip>
  34. <el-select v-model="formInline.priorityLevel" filterable placeholder="请选择" style="width:100%; margin-left: 16px;">
  35. <el-option v-for="item in priorityLevelEnumList" :key="item.code" :label="item.name" :value="item.name" />
  36. </el-select>
  37. </el-form-item>
  38. <el-form-item label="业务线" prop="bizId">
  39. <el-select v-model="formInline.bizId" :disabled="true" filterable placeholder="请选择" style="width:100%;">
  40. <el-option v-for="item in bizIdEnumList" :key="item.code" :label="item.name" :value="item.code" />
  41. </el-select>
  42. </el-form-item>
  43. <el-form-item label="发现阶段" prop="discoveryStage">
  44. <el-select v-model="formInline.discoveryStage" filterable placeholder="请选择" style="width:100%;">
  45. <el-option v-for="item in bugStageEnumList" :key="item.code" :label="item.name" :value="item.code" />
  46. </el-select>
  47. </el-form-item>
  48. <el-form-item label="缺陷类型" prop="theBugType">
  49. <el-cascader v-model="formInline.theBugType" :options="theBugTypeEnumList" :props="{ value:'code', label:'name', children: 'childrenEnums', emitPath: false }" placeholder="请选择" clearable />
  50. </el-form-item>
  51. <el-form-item label="责任人" style="width:100%;" prop="assigner">
  52. <el-select v-model="formInline.assigner" multiple filterable remote :remote-method="remoteMethod" :loading="loading" style="width: 100%" placeholder="请输入姓名或邮箱前缀" @change="getcurrentHandler(formInline.assigner)">
  53. <el-option v-for="item in options" :key="item.idap" :label="item.name" :value="test2(item, 0)">
  54. <div style="display: flex;justify-content: start;">
  55. <div style="min-width:100px;color: #8492a6; font-size: 13px; overflow:hidden">{{ item.deptName }}</div>
  56. <div style="min-width:80px">{{ item.name }}</div>
  57. <div style="min-width:100px;color: #8492a6; font-size: 13px; overflow:hidden">{{ item.idap }}</div>
  58. </div>
  59. </el-option>
  60. </el-select>
  61. </el-form-item>
  62. </div>
  63. <div style="width:100%; margin: 0 4%;">
  64. <el-form-item label="端类型" prop="sysType">
  65. <el-select v-model="formInline.sysType" filterable placeholder="请选择" style="width:100%;">
  66. <el-option v-for="item in sysTypeEnumList" :key="item.code" :label="item.name" :value="item.code" />
  67. </el-select>
  68. </el-form-item>
  69. <el-form-item label="缺陷等级" prop="priority" style="white-space: nowrap;">
  70. <el-tooltip class="item" effect="dark" style="white-space:pre-line;" :content="`P0:阻塞进程的bug,如新功能未实现、app启动失败、如接单crash、开单crash P1:主流程功能bug,新需求的功能性bug建议放在该等级 P2:功能bug,不影响主流程 P3:不影响功能使用的小问题,如界面显示有异常,文案、UI微调,建议优化类`" placement="bottom">
  71. <i style="color: red; margin-left: -17px;" class="el-icon-warning-outline" />
  72. </el-tooltip>
  73. <el-select v-model="formInline.priority" filterable placeholder="请选择" style="width:100%; margin-left: 3px;">
  74. <el-option v-for="item in priorityEnumList" :key="item.code" :label="item.name" :value="item.code" />
  75. </el-select>
  76. </el-form-item>
  77. <el-form-item label="所属模块" prop="moduleIds">
  78. <el-cascader v-model="formInline.moduleIds" collapse-tags :props="props" :options="business_platform_Modular" placeholder="请选择" style="width: 100%" />
  79. </el-form-item>
  80. <el-form-item label="发现方式" prop="discoveryMeth">
  81. <el-select v-model="formInline.discoveryMeth" filterable placeholder="请选择" style="width:100%;">
  82. <el-option v-for="item in discoveryMethEnumList" :key="item.code" :label="item.name" :value="item.code" />
  83. </el-select>
  84. </el-form-item>
  85. <el-form-item label="提报人" style="width:100%;" prop="creatorList">
  86. <el-input v-model="formInline.creatorList" :disabled="true" placeholder="请输入姓名或邮箱前缀" style="width:100%;" />
  87. </el-form-item>
  88. <el-form-item label="修复人" style="width:100%;" prop="currentHandler">
  89. <el-select v-model="formInline.currentHandler" multiple filterable remote :remote-method="remoteMethod" :loading="loading" style="width: 100%" placeholder="请输入姓名或邮箱前缀">
  90. <el-option v-for="item in options" :key="item.idap" :label="item.name" :value="test2(item, 0)">
  91. <div style="display: flex;justify-content: start;">
  92. <div style="min-width:100px;color: #8492a6; font-size: 13px; overflow:hidden">{{ item.deptName }}</div>
  93. <div style="min-width:80px">{{ item.name }}</div>
  94. <div style="min-width:100px;color: #8492a6; font-size: 13px; overflow:hidden">{{ item.idap }}</div>
  95. </div>
  96. </el-option>
  97. </el-select>
  98. </el-form-item>
  99. </div>
  100. </div>
  101. <div class="he" @click="show2 = !show2"><i class="el-icon-arrow-right" :class="{rotateNone:show2,rotate:!show2}" /> 客户端信息</div>
  102. <div v-show="show2" class="Layouts" style="width: 100%;">
  103. <div style="width:100%; margin: 0 4%;height: 200px">
  104. <el-form-item label="客户端">
  105. <el-select v-model="formInline.appId" clearable filterable placeholder="请选择" style="width:100%;" @change="getClient(formInline.appId)">
  106. <el-option v-for="item in appClient" :key="item.code" :label="item.msg" :value="item.code" />
  107. </el-select>
  108. </el-form-item>
  109. <el-form-item label="机型" style="width:100%;">
  110. <el-input v-model="formInline.osType" clearable placeholder="请输入" />
  111. </el-form-item>
  112. <el-form-item label="网络">
  113. <el-select v-model="formInline.networkType" clearable filterable placeholder="请选择" style="width:100%;">
  114. <el-option v-for="item in networkTypeEnumList" :key="item.code" :label="item.name" :value="item.name" />
  115. </el-select>
  116. </el-form-item>
  117. </div>
  118. <div style="width:100%; margin: 0 4%; height: 200px;">
  119. <el-form-item label="影响版本">
  120. <el-select v-model="formInline.appVersion" clearable filterable placeholder="请选择" style="width:100%;">
  121. <el-option v-for="item in Editionlist" :key="item.code" :label="item.msg" :value="item.code" />
  122. </el-select>
  123. </el-form-item>
  124. <el-form-item label="系统版本" style="width:100%;">
  125. <el-input v-model="formInline.sdkVersion" clearable placeholder="请输入" />
  126. </el-form-item>
  127. </div>
  128. </div>
  129. <div style=" margin: 0 4%;">
  130. <el-form-item label="描述" style="width:100%;">
  131. <el-row v-if="editr">
  132. <el-col :span="24">
  133. <!-- <div id="wange" style="background-color: #FFF;" class="toolbar" />
  134. <div id="wange1" class="text" /> -->
  135. <!-- formInline.bugDescribe -->
  136. <normal-area
  137. id="buglist_file_tinymce"
  138. :value.sync="formInline.bugDescribe"
  139. :height="200"
  140. :full-position-style="{ left:'0', bottom: '30px' }"
  141. />
  142. </el-col>
  143. </el-row>
  144. </el-form-item>
  145. <el-form-item label="附件" style="width:100%;">
  146. <section class="upload-main">
  147. <el-tooltip class="item" effect="dark" content="支持的文件格式有:.zip, .xlsx, .txt, .csv, .xls, .mov, .mp4, .m4a, .avi, .amr, .mp3, .wav, .3gpp, .png, .jpg, .jpeg, .gif" placement="top-start" popper-class="tip-style">
  148. <div class="upload-info"><i class="el-icon-question" />格式说明</div>
  149. </el-tooltip>
  150. <el-upload class="upload-demo" action="http://star.xiaojukeji.com/upload/img.node" :on-remove="handleRemove" :on-preview="handlePictureCardPreview" :on-progress="progress" :on-success="handleChange" :on-error="errorUpload" :file-list="fileList" list-type="picture" :before-upload="beforeUpload">
  151. <el-button type="primary">上传附件</el-button>
  152. </el-upload>
  153. </section>
  154. </el-form-item>
  155. </div>
  156. </div>
  157. </div>
  158. <el-form-item style="text-align: right;margin: 15px 0% 0;">
  159. <el-button @click="modalShow = false">取 消</el-button>
  160. <el-button :disabled="dis" type="primary" @click="bug_created(formInline)">创 建</el-button>
  161. </el-form-item>
  162. </el-form>
  163. <el-button id="pasteUpload" type="primary" style="display: none" @click.stop="pasteUpload">upload</el-button>
  164. </el-dialog>
  165. <normal-dialog :show-dialog="showCopyFile" :title="'上传截图'" :width="'40%'" :submit-button="'上传'" :top="'5vh'" @confirm="confirmUpload()" @cancel="showCopyFile=false">
  166. <div class="file-dialog">
  167. <el-form ref="imageForm" label-width="20%" :rules="imageRules" :model="imageName">
  168. <el-form-item label="图片命名" prop="name">
  169. <el-col style="width: 75%">
  170. <el-input v-model="imageName.name" placeholder="请输入图片名称" />
  171. </el-col>
  172. <el-col style="width: 10%">.png</el-col>
  173. </el-form-item>
  174. </el-form>
  175. <div class="image">
  176. <div class="image-center">
  177. <img :src="imageUrl" class="image-url">
  178. </div>
  179. </div>
  180. </div>
  181. </normal-dialog>
  182. <el-dialog title="附件预览" :modal-append-to-body="false" :visible.sync="dialogVisible">
  183. <img width="100%" :src="dialogImageUrl" alt="图片加载失败">
  184. </el-dialog>
  185. </div>
  186. </template>
  187. <script>
  188. const _ = require('lodash')
  189. import { mapGetters } from 'vuex'
  190. import { analysisBizId_id } from '@/utils/crypto-js.js'
  191. // import E from 'wangeditor'
  192. import {
  193. bugGetEnum,
  194. settingGetBizList,
  195. taskListCreate,
  196. releaseList,
  197. bugCreate,
  198. settingQueryBizModuleList
  199. } from '@/api/defectManage'
  200. import { memberQueryMemberInfoByIDAPorName } from '@/api/projectIndex'
  201. import normalDialog from '@/components/dialog/normalDialog'
  202. import '@/views/projectManage/bugList/css/index.css'
  203. import '@/styles/PublicStyle/index.scss'
  204. import normalArea from '@/components/input/normalArea' // 富文本
  205. import 'tinymce/plugins/table'// 插入表格插件
  206. import axios from 'axios'
  207. document.body.onpaste = function(event) {
  208. const data = event.clipboardData || window.clipboardData
  209. const items = data.items
  210. const fileList = [] // 存储文件数据
  211. if (items && items.length) {
  212. // 检索剪切板items
  213. for (let i = 0; i < items.length; i++) {
  214. // console.log(items[i].getAsFile()) // <--- 这里打印出来就就是你想要的文件
  215. fileList.push(items[i].getAsFile())
  216. window.uploadFiles = fileList
  217. }
  218. document.getElementById('pasteUpload').click()
  219. }
  220. }
  221. export default {
  222. name: 'Createdbug',
  223. components: {
  224. normalDialog,
  225. normalArea
  226. },
  227. filters: {
  228. limit(e, limit) {
  229. if (e.length > limit) {
  230. return e.substring(0, limit) + '...'
  231. } else {
  232. return e
  233. }
  234. }
  235. },
  236. props: {
  237. getBugList: {
  238. type: Function,
  239. default: null
  240. },
  241. required: {
  242. type: Number,
  243. default: null
  244. }
  245. },
  246. data() {
  247. return {
  248. nowPageId: -1, // 当前页面的id
  249. dialogImageUrl: '', // 附件展示图片
  250. modalShow: false, // 缺陷新建弹窗
  251. dialogVisible: false,
  252. show2: false,
  253. props: { multiple: true },
  254. dis: false, // 附件上传成功才可以提交
  255. business_platform_Modular: [], // 模块
  256. fileDbList: [], // 附件展示
  257. fileList: [],
  258. taskIdStr: [],
  259. titleName: '',
  260. loading: false,
  261. options: [],
  262. test: [], // 人员查询
  263. form: {},
  264. formInline: {},
  265. userInformation: localStorage.getItem('username'),
  266. userNames: localStorage.getItem('realname'),
  267. rules: {
  268. bizId: [
  269. { required: true, message: '业务线不能为空', trigger: 'change' }
  270. ],
  271. bugName: [
  272. { required: true, message: '标题不能为空', trigger: 'change' }
  273. ],
  274. taskId: [
  275. { required: true, message: '所属任务不能为空', trigger: 'change' }
  276. ],
  277. priorityLevel: [
  278. { required: true, message: '优先级不能为空', trigger: 'change' }
  279. ],
  280. moduleIds: [
  281. { required: true, message: '所属模块不能为空', trigger: 'change' }
  282. ],
  283. discoveryMeth: [
  284. { required: true, message: '发现方式不能为空', trigger: 'change' }
  285. ],
  286. assigner: [
  287. { required: true, message: '责任人不能为空', trigger: 'change' }
  288. ],
  289. sysType: [
  290. { required: true, message: '端类型不能为空', trigger: 'change' }
  291. ],
  292. priority: [
  293. { required: true, message: '缺陷等级不能为空', trigger: 'change' }
  294. ],
  295. discoveryStage: [
  296. { required: true, message: '发现阶段不能为空', trigger: 'change' }
  297. ],
  298. creatorList: [
  299. { required: true, message: '提报人不能为空', trigger: 'change' }
  300. ],
  301. currentHandler: [
  302. { required: true, message: '修复人不能为空', trigger: 'change' }
  303. ]
  304. },
  305. bugEnumList: [], // bug状态
  306. appClient: [], // 客户端
  307. Editionlist: [], // 版本
  308. bizIdEnumList: [], // bug业务线
  309. taskEnumList: [], // 所属任务
  310. bugTypeEnumList: [], // bug类型
  311. priorityEnumList: [], // 缺陷等级
  312. clientTypeEnumList: [], // 版本
  313. repairResultEnumList: [], // 修复结果
  314. theBugTypeEnumList: [], // 缺陷类型
  315. priorityLevelEnumList: [], // 优先级
  316. discoveryMethEnumList: [], // 发现方式
  317. bugStageEnumList: [], // 发现阶段
  318. networkTypeEnumList: [], // 网络
  319. sysTypeEnumList: [], // 端类型
  320. showCopyFile: false, // 复制文件对话框
  321. imageName: { name: null },
  322. imageUrl: null,
  323. editr: false,
  324. formData: [],
  325. formData1: [],
  326. imageRules: {
  327. name: [
  328. { required: true, message: '请输入图片名称', trigger: 'blur' },
  329. { min: 1, max: 50, message: '长度在 1 到 50 个字符', trigger: 'blur' }
  330. ]
  331. }
  332. }
  333. },
  334. computed: {
  335. ...mapGetters(['bizId'])
  336. },
  337. watch: {
  338. bizId: {
  339. handler(newV) {
  340. if (newV === -1) return
  341. this.analysisBizId_id()
  342. this.bugListSelect()
  343. this.bugDataGet()
  344. },
  345. immediate: true
  346. }
  347. },
  348. methods: {
  349. analysisBizId_id() { // 解析路由中的bizId_id
  350. if (!this.$route.query.bizId_id) return
  351. const bizId_id = analysisBizId_id(this.$route.query.bizId_id)
  352. this.nowPageId = bizId_id[1]
  353. },
  354. init(e, ele) {
  355. this.modalShow = true
  356. this.$nextTick(() => {
  357. this.$refs['formInline'].clearValidate()
  358. })
  359. this.fileList = []
  360. this.fileDbList = []
  361. this.formInline = {}
  362. this.$set(
  363. this.formInline,
  364. 'creatorList',
  365. localStorage.getItem('realname')
  366. )
  367. this.$set(this.formInline, 'bizId', this.bizId)
  368. if (e === 1) {
  369. this.titleName = '新建缺陷'
  370. if (ele !== undefined) {
  371. this.$set(this.formInline, 'taskId', ele.id)
  372. }
  373. }
  374. if (e === 2) {
  375. this.titleName = '复制缺陷'
  376. if (ele !== undefined) {
  377. this.formInline = ele
  378. this.show2 = true
  379. this.formInline.creatorList = this.userNames
  380. this.$set(this.formInline, 'bugName', '')
  381. this.$set(this.formInline, 'accessory', '')
  382. this.options = []
  383. this.formData = []
  384. this.formData1 = []
  385. this.formData = this.formInline.assigner.concat(
  386. this.formInline.currentHandler
  387. )
  388. this.formData1 = this.unique6(this.formData)
  389. this.formData1.map(item => {
  390. this.searchUser(item).then(res => {
  391. this.options.push(res.data[0])
  392. })
  393. })
  394. }
  395. }
  396. this.$nextTick(() => {
  397. document.getElementById('soll').scrollTop = 0
  398. this.editr = true
  399. this.getEcharts()
  400. })
  401. },
  402. unique6(arr) {
  403. var newArr = []
  404. newArr = arr.filter(item => {
  405. return newArr.includes(item) ? '' : newArr.push(item)
  406. })
  407. return newArr
  408. },
  409. searchUser(query) {
  410. this.loading = true
  411. return memberQueryMemberInfoByIDAPorName({ memberIDAP: query }).then(
  412. res => {
  413. this.loading = false
  414. return res
  415. }
  416. )
  417. },
  418. getEcharts() {
  419. // setTimeout(() => {
  420. // this.$set(this.formInline, 'bugDescribe', '')
  421. // // const editorRemark = new E('#wange', '#wange1')
  422. // editorRemark.customConfig.menus = [
  423. // 'bold',
  424. // 'italic',
  425. // 'underline',
  426. // 'link',
  427. // 'list',
  428. // 'justify',
  429. // 'table',
  430. // 'foreColor'
  431. // ]
  432. // editorRemark.customConfig.onchange = html => {
  433. // this.formInline.bugDescribe = html
  434. // }
  435. // editorRemark.create()
  436. // }, 100)
  437. },
  438. getcurrentHandler(e) {
  439. this.$set(this.formInline, 'currentHandler', e)
  440. },
  441. getClient(e) {
  442. // 获取版本号
  443. this.$set(this.formInline, 'appVersion', '')
  444. this.Editionlist = this.appClient.filter(
  445. value => value.code === e
  446. )[0].childEnumInfos
  447. },
  448. modalClose() {
  449. this.fileList = []
  450. this.fileDbList = []
  451. this.formInline = {}
  452. this.editr = false
  453. this.modalShow = false
  454. },
  455. // 上传成功回调
  456. handleChange(response, file, fileList) {
  457. const item = { name: file.name, url: 'http:' + file.response.url }
  458. this.fileDbList.push(item)
  459. this.fileList.push(item)
  460. this.formInline.accessory = JSON.stringify(this.fileDbList)
  461. this.dis = false
  462. this.$message({
  463. showClose: true,
  464. message: '文件上传成功',
  465. type: 'success'
  466. })
  467. },
  468. // 上传失败回调
  469. errorUpload() {
  470. this.dis = false
  471. this.$message({
  472. showClose: true,
  473. message: '文件上传失败',
  474. type: 'error'
  475. })
  476. },
  477. handlePictureCardPreview(file) {
  478. // 点击展示附件
  479. this.dialogImageUrl = file.url
  480. this.dialogVisible = true
  481. },
  482. progress(event, file, fileList) {
  483. // 文件上传时的钩子
  484. this.dis = true
  485. },
  486. bug_created(e) {
  487. // 创建(提交)
  488. this.$refs['formInline'].validate(valid => {
  489. if (valid) {
  490. const data = e
  491. data.currentHandler = e.currentHandler.join()
  492. data.assigner = e.assigner.join()
  493. var user = {
  494. name: this.userNames,
  495. ename: this.userInformation,
  496. id: ''
  497. }
  498. bugCreate({ bugBaseInfo: data, user }).then(res => {
  499. this.bugListSelect()
  500. this.modalShow = false
  501. this.$message({
  502. message: res.msg,
  503. type: 'success',
  504. duration: 1000,
  505. offset: 150
  506. })
  507. this.$emit('father')
  508. if (this.getBugList) {
  509. this.getBugList()
  510. }
  511. this.$emit('getBugList')
  512. })
  513. }
  514. })
  515. },
  516. test2(item, e) {
  517. // 获取团队人员信息
  518. if (typeof this.test[item.idap] === 'undefined') {
  519. item.role = e
  520. this.test[item.idap] = item
  521. }
  522. return item.idap
  523. },
  524. remoteMethod(query) {
  525. // 人员查询
  526. if (query !== '') {
  527. this.loading = true
  528. setTimeout(() => {
  529. this.loading = false
  530. memberQueryMemberInfoByIDAPorName({ memberIDAP: query }).then(res => {
  531. const obj = {}
  532. this.options = res.data.reduce((cur, next) => {
  533. obj[next.idap] ? '' : (obj[next.idap] = true && cur.push(next))
  534. return cur
  535. }, [])
  536. })
  537. }, 200)
  538. } else {
  539. this.options = []
  540. }
  541. },
  542. async getTaskList(val) {
  543. // 获取所属任务列表
  544. const params = {
  545. bizId: this.bizId
  546. }
  547. switch (this.$route.name) {
  548. case '项目详情':
  549. params.projectId = this.nowPageId
  550. break
  551. case '需求详情':
  552. params.requireId = this.nowPageId
  553. break
  554. case '任务详情':
  555. params.id = this.nowPageId
  556. break
  557. }
  558. if (val) {
  559. params.name = val
  560. }
  561. const res = await taskListCreate(params)
  562. if (res.code === 200) {
  563. this.taskEnumList = res.data || []
  564. }
  565. },
  566. debounceQuery: _.debounce(function() {
  567. this.getTaskList(...arguments)
  568. }, 500),
  569. bugListSelect() {
  570. this.$set(this.formInline, 'creatorList', this.userNames)
  571. settingGetBizList({}).then(res => {
  572. this.bizIdEnumList = res.data // biz
  573. })
  574. this.getTaskList()
  575. bugGetEnum().then(res => {
  576. this.bugEnumList = res.data.bugEnumList // status
  577. this.bugTypeEnumList = res.data.bugTypeEnumList // bug类型
  578. this.bugStageEnumList = res.data.bugStageEnumList // 发现阶段
  579. this.sysTypeEnumList = res.data.sysTypeEnumList // 端类型
  580. this.discoveryMethEnumList = res.data.discoveryMethEnumList // 发现方式
  581. this.priorityEnumList = res.data.priorityEnumList // 缺陷等级
  582. this.priorityLevelEnumList = res.data.priorityLevelEnumList // 优先级
  583. this.repairResultEnumList = res.data.repairResultEnumList // 修复结果
  584. this.clientTypeEnumList = res.data.clientTypeEnumList // 版本
  585. this.sysTypeEnumList = res.data.sysTypeEnumList // 客户端
  586. this.networkTypeEnumList = res.data.networkTypeEnumList // 网络
  587. this.theBugTypeEnumList = this.deleteChild(res.data.theBugTypeEnumList) // 缺陷类型
  588. })
  589. releaseList().then(res => {
  590. this.appClient = res.data.appClient // 客户端
  591. })
  592. },
  593. deleteChild(arr) {
  594. // 删除无用子属性
  595. const bfs = arr => {
  596. arr.forEach(item => {
  597. if (!item.childrenEnums || item.childrenEnums.length === 0) {
  598. delete item.childrenEnums
  599. } else {
  600. this.deleteChild(item.childrenEnums)
  601. }
  602. })
  603. }
  604. bfs(arr)
  605. return arr
  606. },
  607. bugDataGet() {
  608. // 所属模块
  609. settingQueryBizModuleList(this.bizId).then(
  610. res => {
  611. this.business_platform_Modular = res.data.map(item => ({
  612. ...item,
  613. value: item.id,
  614. label: item.moduleName,
  615. children:
  616. item.childModules.length === 0
  617. ? null
  618. : item.childModules.map(item1 => ({
  619. ...item1,
  620. value: item1.id,
  621. label: item1.moduleName,
  622. children:
  623. item1.childModules.length === 0
  624. ? null
  625. : item1.childModules.map(item2 => ({
  626. ...item2,
  627. value: item2.id,
  628. label: item2.moduleName
  629. }))
  630. }))
  631. }))
  632. }
  633. )
  634. },
  635. beforeUpload(file) {
  636. const reg = new RegExp(
  637. /.*(zip|xlsx|text|csv|xls|mov|mp4|m4a|avi|amr|mp3|wav|3gpp|gif|jpeg|png|jpg)/i
  638. )
  639. const isUpload = file.type.match(reg)
  640. if (isUpload === false) {
  641. this.$message({
  642. message: '不支持上传此文件格式',
  643. type: 'warning'
  644. })
  645. return false
  646. }
  647. const isLt200M = file.size / 1024 / 1024 < 20
  648. if (!isLt200M) {
  649. this.$message({
  650. message: '上传文件大小不能超过 10MB!',
  651. type: 'warning'
  652. })
  653. return false
  654. }
  655. },
  656. handleRemove(file, fileList) {
  657. this.fileDbList = this.fileDbList.filter(item => {
  658. return item.name !== file.name
  659. })
  660. this.fileList = this.fileList.filter(item => {
  661. return item.name !== file.name
  662. })
  663. },
  664. generateMixed(len) {
  665. const chars = [
  666. '0',
  667. '1',
  668. '2',
  669. '3',
  670. '4',
  671. '5',
  672. '6',
  673. '7',
  674. '8',
  675. '9',
  676. 'A',
  677. 'B',
  678. 'C',
  679. 'D',
  680. 'E',
  681. 'F',
  682. 'G',
  683. 'H',
  684. 'I',
  685. 'J',
  686. 'K',
  687. 'L',
  688. 'M',
  689. 'N',
  690. 'O',
  691. 'P',
  692. 'Q',
  693. 'R',
  694. 'S',
  695. 'T',
  696. 'U',
  697. 'V',
  698. 'W',
  699. 'X',
  700. 'Y',
  701. 'Z'
  702. ]
  703. let res = ''
  704. for (let i = 0; i < len; i++) {
  705. const id = Math.ceil(Math.random() * 35)
  706. res += chars[id]
  707. }
  708. return res
  709. },
  710. pasteUpload() {
  711. if (!this.modalShow) {
  712. return false
  713. }
  714. if (window.uploadFiles[0]) {
  715. const reader = new FileReader()
  716. reader.readAsDataURL(window.uploadFiles[0])
  717. reader.onload = () => {
  718. const reg = new RegExp(/image\/png/)
  719. this.imageUrl = reader.result
  720. if (this.imageUrl.match(reg)) {
  721. // 判断是否是图片
  722. this.showCopyFile = true
  723. this.imageName.name = this.generateMixed(10)
  724. }
  725. }
  726. this.showCopyFile = true
  727. }
  728. },
  729. async confirmUpload() {
  730. if (
  731. this.imageName.name === null ||
  732. this.imageName.name.replace(/\s+/g, '') === ''
  733. ) {
  734. return false
  735. }
  736. const isExist = this.fileList.some(item => {
  737. return this.imageName.name === item.name
  738. })
  739. if (isExist) {
  740. this.$message({
  741. showClose: true,
  742. message: '图片名称重复',
  743. type: 'error'
  744. })
  745. return false
  746. }
  747. this.showCopyFile = false
  748. const res = await this.updateFile(window.uploadFiles[0])
  749. const data = res.data
  750. const item = {
  751. name: `${this.imageName.name}.png` || `${this.generateMixed(10)}.png`,
  752. status: 'success',
  753. url: 'http:' + data.url
  754. }
  755. this.fileList.push(item)
  756. this.fileDbList.push(item)
  757. this.formInline.accessory = JSON.stringify(this.fileDbList)
  758. this.$message({
  759. showClose: true,
  760. message: '文件上传成功',
  761. type: 'success'
  762. })
  763. this.imageName.name = null
  764. this.imageUrl = null
  765. window.uploadFiles = null
  766. },
  767. updateFile(file) {
  768. const param = new FormData() // 创建form对象
  769. param.append('file', file) // 通过append向form对象添加数据
  770. const config = {
  771. headers: {
  772. 'Content-Type': 'multipart/form-data'
  773. },
  774. withCredentials: false
  775. } // 添加请求头
  776. return new Promise((resolve, reject) => {
  777. axios
  778. .post('http://star.xiaojukeji.com/upload/img.node', param, config)
  779. .then(response => {
  780. resolve(response)
  781. })
  782. .catch(err => {
  783. reject(err)
  784. })
  785. })
  786. }
  787. }
  788. }
  789. </script>
  790. <style lang="scss" scoped>
  791. .file-dialog {
  792. display: flex;
  793. flex-direction: column;
  794. align-items: center;
  795. .el-form {
  796. width: 100%;
  797. }
  798. .image {
  799. position: relative;
  800. width: 61%;
  801. padding-top: 60%;
  802. border: 1px solid #409eff;
  803. border-radius: 4px;
  804. .image-center {
  805. padding: 1%;
  806. position: absolute;
  807. top: 0;
  808. left: 0;
  809. width: 100%;
  810. height: 100%;
  811. overflow-x: hidden;
  812. display: flex;
  813. justify-content: center;
  814. }
  815. .image-url {
  816. height: 100%;
  817. }
  818. }
  819. }
  820. .upload-main {
  821. position: relative;
  822. }
  823. .upload-info {
  824. position: absolute;
  825. left: 110px;
  826. top: 50%;
  827. transform: translateY(-50%);
  828. color: #409eff;
  829. font-size: 12px;
  830. i {
  831. margin-right: 5px;
  832. }
  833. }
  834. .belong-task {
  835. max-width: 500px;
  836. display: flex;
  837. .modules-name {
  838. width: calc(100% - 100px);
  839. overflow: hidden;
  840. text-overflow: ellipsis;
  841. white-space: nowrap;
  842. }
  843. .modules {
  844. color: #999999;
  845. }
  846. .task-id {
  847. color: #999999;
  848. width: 80px;
  849. margin-right: 20px;
  850. }
  851. .name {
  852. color: #333333;
  853. margin-right: 20px;
  854. }
  855. }
  856. </style>