createdBug.vue 29 KB

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