createdBug.vue 33 KB

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