createdBug.vue 31 KB

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