createdBug.vue 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400
  1. <template>
  2. <div class="Parent">
  3. <el-dialog :visible.sync="modalShow" width="60%" top="4vh" :close-on-click-modal="false" @close="modalClose">
  4. <el-form ref="formInline" label-position="left" :model="formInline" :rules="rules" label-width="90px">
  5. <el-row style="margin: 0 4% 2% 4%;">
  6. <el-col :span="24">
  7. <div class="flex_start titFont">
  8. <div class="titIcon" />
  9. <span style="margin-left: 1%; width: 100px;">{{ titleName }}</span>
  10. </div>
  11. </el-col>
  12. </el-row>
  13. <div style="height: 500px; overflow:scroll; overflow-x: hidden">
  14. <div>
  15. <el-row style="margin: 0 4%;">
  16. <el-col :span="24">
  17. <el-form-item label="标题" prop="bugName">
  18. <el-input v-model="formInline.bugName" size="small" placeholder="请输入" style="width:100%;" />
  19. </el-form-item>
  20. </el-col>
  21. </el-row>
  22. <div class="Layouts" style="width: 100%;">
  23. <div style="width:100%; margin: 0 4%;">
  24. <el-form-item label="所属任务" prop="taskId">
  25. <el-select v-model="formInline.taskId" size="small" filterable placeholder="请选择" style="width:100%;" @click.native="bugListSelect">
  26. <el-option v-for="item in taskEnumList" :key="item.id" :label="item.name" :value="item.id" />
  27. </el-select>
  28. </el-form-item>
  29. <el-form-item label="优先级" prop="priorityLevel">
  30. <el-tooltip
  31. class="item"
  32. effect="dark"
  33. content="i. High(阻塞阻塞进程的bug、主流程的严重bug,如app启动失败、接单crash,当天必须立即修复) ii. Medium(功bug,新需求的功能性bug建议放在该等级,可协商修复期限) iv. Low(不影响功能使用的小问题,如界面显示有异常,文案、UI微调,可协商修复期限)"
  34. placement="bottom"
  35. >
  36. <i style="color: red; margin-left: -31px;" class="el-icon-warning-outline" />
  37. </el-tooltip>
  38. <el-select v-model="formInline.priorityLevel" size="small" filterable placeholder="请选择" style="width:100%; margin-left: 12px;">
  39. <el-option v-for="item in priorityLevelEnumList" :key="item.code" :label="item.name" :value="item.name" />
  40. </el-select>
  41. </el-form-item>
  42. <el-form-item label="业务线" prop="bizId">
  43. <el-select v-model="formInline.bizId" :disabled="true" size="small" filterable placeholder="请选择" style="width:100%;">
  44. <el-option v-for="item in bizIdEnumList" :key="item.code" :label="item.name" :value="item.code" />
  45. </el-select>
  46. </el-form-item>
  47. <el-form-item label="发现阶段" prop="discoveryStage">
  48. <el-select v-model="formInline.discoveryStage" size="small" filterable placeholder="请选择" style="width:100%;">
  49. <el-option v-for="item in bugStageEnumList" :key="item.code" :label="item.name" :value="item.code" />
  50. </el-select>
  51. </el-form-item>
  52. <el-form-item label="缺陷类型" prop="theBugType">
  53. <el-select v-model="formInline.theBugType" size="small" filterable placeholder="请选择" style="width:100%;">
  54. <el-option v-for="item in theBugTypeEnumList" :key="item.code" :label="item.name" :value="item.code" />
  55. </el-select>
  56. </el-form-item>
  57. <el-form-item label="责任人" style="width:100%;" prop="assigner">
  58. <el-select v-model="formInline.assigner" filterable remote reserve-keyword :remote-method="remoteMethod" :loading="loading" size="small" style="width: 100%" placeholder="请输入姓名或邮箱前缀" @change="getcurrentHandler(formInline.assigner)">
  59. <el-option v-for="item in options" :key="item.idap" :label="item.name" :value="test2(item, 0)">
  60. <div style="display: flex;justify-content: start;">
  61. <div style="min-width:100px;color: #8492a6; font-size: 13px; overflow:hidden">{{ item.deptName }}</div>
  62. <div style="min-width:80px">{{ item.name }}</div>
  63. <div style="min-width:100px;color: #8492a6; font-size: 13px; overflow:hidden">{{ item.idap }}</div>
  64. </div>
  65. </el-option>
  66. </el-select>
  67. </el-form-item>
  68. </div>
  69. <div style="width:100%; margin: 0 4%;">
  70. <el-form-item label="端类型" prop="sysType">
  71. <el-select v-model="formInline.sysType" size="small" filterable placeholder="请选择" style="width:100%;">
  72. <el-option v-for="item in sysTypeEnumList" :key="item.code" :label="item.name" :value="item.code" />
  73. </el-select>
  74. </el-form-item>
  75. <el-form-item label="缺陷等级" prop="priority">
  76. <el-tooltip
  77. class="item"
  78. effect="dark"
  79. style="white-space:pre-line;"
  80. :content="dd"
  81. placement="bottom"
  82. >
  83. <i style="color: red; margin-left: -17px;" class="el-icon-warning-outline" />
  84. </el-tooltip>
  85. <el-select v-model="formInline.priority" size="small" filterable placeholder="请选择" style="width:100%; margin-left: -2px;">
  86. <el-option v-for="item in priorityEnumList" :key="item.code" :label="item.name" :value="item.code" />
  87. </el-select>
  88. </el-form-item>
  89. <el-form-item label="所属模块" prop="moduleIds">
  90. <el-cascader v-model="formInline.moduleIds" size="small" collapse-tags :props="props" :options="business_platform_Modular" placeholder="请选择" style="width: 100%" />
  91. </el-form-item>
  92. <el-form-item label="发现方式" prop="discoveryMeth">
  93. <el-select v-model="formInline.discoveryMeth" size="small" filterable placeholder="请选择" style="width:100%;">
  94. <el-option v-for="item in discoveryMethEnumList" :key="item.code" :label="item.name" :value="item.code" />
  95. </el-select>
  96. </el-form-item>
  97. <el-form-item label="提报人" style="width:100%;" prop="currentHandler">
  98. <el-input v-model="formInline.creatorList" :disabled="true" size="small" placeholder="请输入姓名或邮箱前缀" style="width:100%;" />
  99. </el-form-item>
  100. <el-form-item label="修复人" style="width:100%;" prop="currentHandler">
  101. <el-select v-model="formInline.currentHandler" filterable remote reserve-keyword :remote-method="remoteMethod" :loading="loading" size="small" style="width: 100%" placeholder="请输入姓名或邮箱前缀">
  102. <el-option v-for="item in options" :key="item.idap" :label="item.name" :value="test2(item, 0)">
  103. <div style="display: flex;justify-content: start;">
  104. <div style="min-width:100px;color: #8492a6; font-size: 13px; overflow:hidden">{{ item.deptName }}</div>
  105. <div style="min-width:80px">{{ item.name }}</div>
  106. <div style="min-width:100px;color: #8492a6; font-size: 13px; overflow:hidden">{{ item.idap }}</div>
  107. </div>
  108. </el-option>
  109. </el-select>
  110. </el-form-item>
  111. </div>
  112. </div>
  113. <div class="he" @click="show2 = !show2"><i class="el-icon-arrow-right" :class="{rotateNone:show2,rotate:!show2}" /> 客户端信息</div>
  114. <div v-show="show2" class="Layouts" style="width: 100%;">
  115. <div style="width:100%; margin: 0 4%;height: 200px">
  116. <el-form-item label="客户端">
  117. <el-select v-model="formInline.appId" size="small" clearable filterable placeholder="请选择" style="width:100%;" @change="getClient(formInline.appId)">
  118. <el-option v-for="item in appClient" :key="item.code" :label="item.msg" :value="item.code" />
  119. </el-select>
  120. </el-form-item>
  121. <el-form-item label="机型" style="width:100%;">
  122. <el-input v-model="formInline.osType" size="small" clearable placeholder="请输入" />
  123. </el-form-item>
  124. <el-form-item label="网络">
  125. <el-select v-model="formInline.networkType" size="small" clearable filterable placeholder="请选择" style="width:100%;">
  126. <el-option v-for="item in networkTypeEnumList" :key="item.code" :label="item.name" :value="item.name" />
  127. </el-select>
  128. </el-form-item>
  129. </div>
  130. <div style="width:100%; margin: 0 4%; height: 200px;">
  131. <el-form-item label="影响版本">
  132. <el-select v-model="formInline.appVersion" size="small" clearable filterable placeholder="请选择" style="width:100%;">
  133. <el-option v-for="item in Editionlist" :key="item.code" :label="item.msg" :value="item.code" />
  134. </el-select>
  135. </el-form-item>
  136. <el-form-item label="系统版本" style="width:100%;">
  137. <el-input v-model="formInline.sdkVersion" clearable placeholder="请输入" size="small" />
  138. </el-form-item>
  139. </div>
  140. </div>
  141. <div style=" margin: 0 4%;">
  142. <el-form-item label="描述" style="width:100%;">
  143. <el-row>
  144. <el-col :span="24">
  145. <div id="wange" style="background-color: #FFF;" class="toolbar" />
  146. <div id="wange1" class="text" />
  147. </el-col>
  148. </el-row>
  149. </el-form-item>
  150. <el-form-item label="附件" style="width:100%;">
  151. <el-upload
  152. class="upload-demo"
  153. action="http://star.xiaojukeji.com/upload/img.node"
  154. :on-preview="handlePictureCardPreview"
  155. :on-progress="progress"
  156. :on-success="handleChange"
  157. :file-list="fileList"
  158. list-type="picture"
  159. >
  160. <el-button size="small" type="primary">上传附件</el-button>
  161. </el-upload>
  162. </el-form-item>
  163. </div>
  164. </div>
  165. </div>
  166. <el-form-item style="text-align: right;margin: 15px 4% 0 4%;">
  167. <el-button size="small" @click="modalShow = false">取 消</el-button>
  168. <el-button :disabled="dis" type="primary" size="small" @click="bug_created(formInline)">创 建</el-button>
  169. </el-form-item>
  170. </el-form>
  171. </el-dialog>
  172. <el-dialog title="附件预览" :visible.sync="dialogVisible">
  173. <img width="100%" :src="dialogImageUrl" alt="图片加载失败">
  174. </el-dialog>
  175. </div>
  176. </template>
  177. <script>
  178. import E from 'wangeditor'
  179. import { bugGetEnum, settingGetBizList, taskListCreate, releaseList, bugCreate, settingQueryBizModuleList } from '@/api/defectManage'
  180. import { memberQueryMemberInfoByIDAPorName } from '@/api/projectIndex'
  181. import '@/views/projectManage/bugList/css/index.css'
  182. export default {
  183. name: 'Createdbug',
  184. props: {
  185. getBugList: {
  186. type: Function,
  187. default: null
  188. }
  189. },
  190. data() {
  191. return {
  192. dd: `P0:阻塞进程的bug,如新功能未实现、app启动失败、如接单crash、开单crash
  193. P1:主流程功能bug,新需求的功能性bug建议放在该等级
  194. P2:功能bug,不影响主流程
  195. P3:不影响功能使用的小问题,如界面显示有异常,文案、UI微调,建议优化类`,
  196. dialogImageUrl: '', // 附件展示图片
  197. modalShow: false, // 缺陷新建弹窗
  198. dialogVisible: false,
  199. show2: false,
  200. props: { multiple: true },
  201. dis: false, // 附件上传成功才可以提交
  202. business_platform_Modular: [], // 模块
  203. fileDbList: [], // 附件展示
  204. editorRemark: '',
  205. fileList: [],
  206. taskIdStr: [],
  207. titleName: '',
  208. loading: false,
  209. options: [],
  210. test: [], // 人员查询
  211. form: {},
  212. formInline: {},
  213. userInformation: localStorage.getItem('username'),
  214. userNames: localStorage.getItem('realname'),
  215. rules: {
  216. bizId: [{ required: true, message: '业务线不能为空', trigger: 'change' }],
  217. bugName: [{ required: true, message: '标题不能为空', trigger: 'change' }],
  218. taskId: [{ required: true, message: '所属任务不能为空', trigger: 'change' }],
  219. priorityLevel: [{ required: true, message: '优先级不能为空', trigger: 'change' }],
  220. moduleIds: [{ required: true, message: '所属模块不能为空', trigger: 'change' }],
  221. discoveryMeth: [{ required: true, message: '发现方式不能为空', trigger: 'change' }],
  222. assigner: [{ required: true, message: '责任人不能为空', trigger: 'change' }],
  223. sysType: [{ required: true, message: '端类型不能为空', trigger: 'change' }],
  224. priority: [{ required: true, message: '缺陷等级不能为空', trigger: 'change' }],
  225. discoveryStage: [{ required: true, message: '发现阶段不能为空', trigger: 'change' }],
  226. theBugType: [{ required: true, message: '缺陷类型不能为空', trigger: 'change' }],
  227. currentHandler: [{ required: true, message: '修复人不能为空', trigger: 'change' }]
  228. },
  229. bugEnumList: [], // bug状态
  230. appClient: [], // 客户端
  231. Editionlist: [], // 版本
  232. bizIdEnumList: [], // bug业务线
  233. taskEnumList: [], // 所属任务
  234. bugTypeEnumList: [], // bug类型
  235. priorityEnumList: [], // 缺陷等级
  236. clientTypeEnumList: [], // 版本
  237. repairResultEnumList: [], // 修复结果
  238. theBugTypeEnumList: [], // 缺陷类型
  239. priorityLevelEnumList: [], // 优先级
  240. discoveryMethEnumList: [], // 发现方式
  241. bugStageEnumList: [], // 发现阶段
  242. networkTypeEnumList: [], // 网络
  243. sysTypeEnumList: [] // 端类型
  244. }
  245. },
  246. created() {
  247. this.bugListSelect()
  248. this.bugDataGet()
  249. },
  250. mounted() {
  251. this.form.remark = 'dd'
  252. setTimeout(() => {
  253. this.$set(this.formInline, 'bugDescribe', '')
  254. this.editorRemark = new E('#wange', '#wange1')
  255. this.editorRemark.customConfig.menus = ['bold', 'italic', 'underline', 'link', 'list', 'justify', 'table', 'foreColor']
  256. this.editorRemark.customConfig.onchange = (html) => {
  257. this.formInline.bugDescribe = html
  258. }
  259. this.editorRemark.create()
  260. }, 100)
  261. },
  262. methods: {
  263. init(e, ele) {
  264. this.modalShow = true
  265. this.$nextTick(() => {
  266. this.$refs['formInline'].clearValidate()
  267. })
  268. this.fileList = []
  269. this.$set(this.formInline, 'creatorList', this.userNames)
  270. this.$set(this.formInline, 'bizId', Number(localStorage.getItem('bizId')))
  271. if (e === 1) {
  272. this.titleName = '新建缺陷'
  273. if (ele !== undefined) {
  274. this.$set(this.formInline, 'taskId', ele.id)
  275. }
  276. }
  277. },
  278. getcurrentHandler(e) {
  279. this.$set(this.formInline, 'currentHandler', e)
  280. },
  281. getClient(e) { // 获取版本号
  282. this.$set(this.formInline, 'appVersion', '')
  283. this.Editionlist = this.appClient.filter(value => value.code === e)[0].childEnumInfos
  284. },
  285. modalClose() {
  286. this.formInline = {}
  287. this.editorRemark.txt.clear()
  288. this.modalShow = false
  289. },
  290. // 上传成功回调
  291. handleChange(response, file, fileList) {
  292. this.fileDbList.push({ 'name': file.name, 'url': 'http:' + file.response.url })
  293. this.formInline.accessory = JSON.stringify(this.fileDbList)
  294. this.dis = false
  295. this.$message({ showClose: true, message: '文件上传成功', type: 'success' })
  296. },
  297. handlePictureCardPreview(file) { // 点击展示附件
  298. this.dialogImageUrl = file.url
  299. this.dialogVisible = true
  300. },
  301. progress(event, file, fileList) { // 文件上传时的钩子
  302. this.dis = true
  303. },
  304. bug_created(e) { // 创建(提交)
  305. this.$refs['formInline'].validate((valid) => {
  306. if (valid) {
  307. var user = { name: this.userNames, ename: this.userInformation, id: '' }
  308. bugCreate({ bugBaseInfo: e, user }).then(res => {
  309. this.bugListSelect()
  310. this.modalShow = false
  311. this.$message({ message: res.msg, type: 'success', duration: 1000, offset: 150 })
  312. if (this.getBugList) {
  313. this.getBugList()
  314. }
  315. })
  316. }
  317. })
  318. },
  319. test2(item, e) { // 获取团队人员信息
  320. if (typeof this.test[item.idap] === 'undefined') {
  321. item.role = e
  322. this.test[item.idap] = item
  323. }
  324. return item.idap
  325. },
  326. remoteMethod(query) {
  327. // 人员查询
  328. if (query !== '') {
  329. this.loading = true
  330. setTimeout(() => {
  331. this.loading = false
  332. memberQueryMemberInfoByIDAPorName({ memberIDAP: query }).then(res => {
  333. const obj = {}
  334. this.options = res.data.reduce((cur, next) => {
  335. obj[next.idap] ? '' : obj[next.idap] = true && cur.push(next)
  336. return cur
  337. }, [])
  338. })
  339. }, 200)
  340. } else {
  341. this.options = []
  342. }
  343. },
  344. bugListSelect() {
  345. this.$set(this.formInline, 'creatorList', this.userNames)
  346. settingGetBizList({}).then(res => {
  347. this.bizIdEnumList = res.data // biz
  348. })
  349. taskListCreate({ bizId: Number(localStorage.getItem('bizId')) }).then(res => {
  350. this.taskEnumList = res.data // 所属任务
  351. })
  352. bugGetEnum().then(res => {
  353. this.bugEnumList = res.data.bugEnumList // status
  354. this.bugTypeEnumList = res.data.bugTypeEnumList // bug类型
  355. this.bugStageEnumList = res.data.bugStageEnumList // 发现阶段
  356. this.sysTypeEnumList = res.data.sysTypeEnumList // 端类型
  357. this.discoveryMethEnumList = res.data.discoveryMethEnumList // 发现方式
  358. this.priorityEnumList = res.data.priorityEnumList // 缺陷等级
  359. this.priorityLevelEnumList = res.data.priorityLevelEnumList // 优先级
  360. this.repairResultEnumList = res.data.repairResultEnumList // 修复结果
  361. this.clientTypeEnumList = res.data.clientTypeEnumList // 版本
  362. this.sysTypeEnumList = res.data.sysTypeEnumList // 客户端
  363. this.networkTypeEnumList = res.data.networkTypeEnumList // 网络
  364. this.theBugTypeEnumList = res.data.theBugTypeEnumList // 缺陷类型
  365. })
  366. releaseList().then(res => {
  367. this.appClient = res.data.appClient // 客户端
  368. })
  369. },
  370. bugDataGet() { // 所属模块
  371. settingQueryBizModuleList(Number(localStorage.getItem('bizId'))).then(res => {
  372. this.business_platform_Modular = res.data.map(item => ({
  373. ...item,
  374. value: item.id,
  375. label: item.moduleName,
  376. children: item.childModules.length === 0 ? null : item.childModules.map(item1 => ({
  377. ...item1,
  378. value: item1.id,
  379. label: item1.moduleName,
  380. children: item1.childModules.length === 0 ? null : item1.childModules.map(item2 => ({
  381. ...item2,
  382. value: item2.id,
  383. label: item2.moduleName
  384. }))
  385. }))
  386. }))
  387. })
  388. }
  389. }
  390. }
  391. </script>