reafactor: markdown 替换富文本框
This commit is contained in:
parent
9b884c168b
commit
4e677b5e3b
|
@ -415,7 +415,8 @@
|
||||||
|
|
||||||
<select id="listForMinder" resultType="io.metersphere.base.domain.TestCaseWithBLOBs">
|
<select id="listForMinder" resultType="io.metersphere.base.domain.TestCaseWithBLOBs">
|
||||||
select
|
select
|
||||||
id, `name`, node_id, node_path, `type`, `method`, maintainer, priority, prerequisite, remark, steps
|
<include refid="io.metersphere.base.mapper.TestCaseMapper.Base_Column_List"/>,
|
||||||
|
<include refid="io.metersphere.base.mapper.TestCaseMapper.Blob_Column_List"/>
|
||||||
from test_case
|
from test_case
|
||||||
<include refid="queryWhereCondition"/>
|
<include refid="queryWhereCondition"/>
|
||||||
<include refid="io.metersphere.base.mapper.ext.ExtBaseMapper.orders"/>
|
<include refid="io.metersphere.base.mapper.ext.ExtBaseMapper.orders"/>
|
||||||
|
|
|
@ -4,9 +4,17 @@ import io.metersphere.commons.exception.MSException;
|
||||||
import io.metersphere.i18n.Translator;
|
import io.metersphere.i18n.Translator;
|
||||||
import org.apache.commons.collections.CollectionUtils;
|
import org.apache.commons.collections.CollectionUtils;
|
||||||
import org.aspectj.util.FileUtil;
|
import org.aspectj.util.FileUtil;
|
||||||
|
import org.springframework.core.io.FileSystemResource;
|
||||||
|
import org.springframework.http.HttpHeaders;
|
||||||
|
import org.springframework.http.MediaType;
|
||||||
|
import org.springframework.http.ResponseEntity;
|
||||||
import org.springframework.web.multipart.MultipartFile;
|
import org.springframework.web.multipart.MultipartFile;
|
||||||
|
|
||||||
|
import javax.servlet.ServletOutputStream;
|
||||||
|
import javax.servlet.http.HttpServletResponse;
|
||||||
import java.io.*;
|
import java.io.*;
|
||||||
|
import java.net.URLEncoder;
|
||||||
|
import java.util.Date;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
public class FileUtils {
|
public class FileUtils {
|
||||||
|
@ -31,4 +39,36 @@ public class FileUtils {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static String uploadFile(MultipartFile uploadFile, String path, String name) {
|
||||||
|
if (uploadFile == null) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
File testDir = new File(path);
|
||||||
|
if (!testDir.exists()) {
|
||||||
|
testDir.mkdirs();
|
||||||
|
}
|
||||||
|
String filePath = testDir + "/" + name;
|
||||||
|
File file = new File(filePath);
|
||||||
|
try (InputStream in = uploadFile.getInputStream(); OutputStream out = new FileOutputStream(file)) {
|
||||||
|
file.createNewFile();
|
||||||
|
FileUtil.copyStream(in, out);
|
||||||
|
} catch (IOException e) {
|
||||||
|
LogUtil.error(e.getMessage(), e);
|
||||||
|
MSException.throwException(Translator.get("upload_fail"));
|
||||||
|
}
|
||||||
|
return filePath;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static String uploadFile(MultipartFile uploadFile, String path) {
|
||||||
|
return uploadFile(uploadFile, path, uploadFile.getOriginalFilename());
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void deleteFile(String path) {
|
||||||
|
File file = new File(path);
|
||||||
|
if (file.exists()) {
|
||||||
|
file.delete();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -60,6 +60,7 @@ public class ShiroUtils {
|
||||||
filterChainDefinitionMap.put("/document", "apikey, authc"); // 跳转到 /document 不用校验 csrf
|
filterChainDefinitionMap.put("/document", "apikey, authc"); // 跳转到 /document 不用校验 csrf
|
||||||
filterChainDefinitionMap.put("/test/case/file/preview/**", "apikey, authc"); // 预览测试用例附件 不用校验 csrf
|
filterChainDefinitionMap.put("/test/case/file/preview/**", "apikey, authc"); // 预览测试用例附件 不用校验 csrf
|
||||||
filterChainDefinitionMap.put("/mock", "apikey, authc"); // 跳转到 /mock接口 不用校验 csrf
|
filterChainDefinitionMap.put("/mock", "apikey, authc"); // 跳转到 /mock接口 不用校验 csrf
|
||||||
|
filterChainDefinitionMap.put("/resource/md/get/**", "apikey, authc");
|
||||||
}
|
}
|
||||||
|
|
||||||
public static Cookie getSessionIdCookie(){
|
public static Cookie getSessionIdCookie(){
|
||||||
|
|
|
@ -0,0 +1,39 @@
|
||||||
|
package io.metersphere.controller;
|
||||||
|
|
||||||
|
import io.metersphere.commons.constants.RoleConstants;
|
||||||
|
import io.metersphere.controller.request.MdUploadRequest;
|
||||||
|
import io.metersphere.service.ResourceService;
|
||||||
|
import org.apache.shiro.authz.annotation.Logical;
|
||||||
|
import org.apache.shiro.authz.annotation.RequiresRoles;
|
||||||
|
import org.springframework.core.io.FileSystemResource;
|
||||||
|
import org.springframework.http.ResponseEntity;
|
||||||
|
import org.springframework.web.bind.annotation.*;
|
||||||
|
import org.springframework.web.multipart.MultipartFile;
|
||||||
|
|
||||||
|
import javax.annotation.Resource;
|
||||||
|
|
||||||
|
@RestController
|
||||||
|
@RequestMapping(value = "/resource")
|
||||||
|
@RequiresRoles(value = {RoleConstants.TEST_MANAGER, RoleConstants.TEST_USER, RoleConstants.TEST_VIEWER}, logical = Logical.OR)
|
||||||
|
public class ResourceController {
|
||||||
|
|
||||||
|
@Resource
|
||||||
|
ResourceService resourceService;
|
||||||
|
@PostMapping(value = "/md/upload", consumes = {"multipart/form-data"})
|
||||||
|
@RequiresRoles(value = {RoleConstants.TEST_MANAGER, RoleConstants.TEST_USER,}, logical = Logical.OR)
|
||||||
|
public void upload(@RequestPart(value = "request") MdUploadRequest request, @RequestPart(value = "file") MultipartFile file) {
|
||||||
|
resourceService.mdUpload(request, file);
|
||||||
|
}
|
||||||
|
|
||||||
|
@GetMapping(value = "/md/get/{fileName}")
|
||||||
|
public ResponseEntity<FileSystemResource> getFile(@PathVariable("fileName") String fileName) {
|
||||||
|
return resourceService.getMdImage(fileName);
|
||||||
|
}
|
||||||
|
|
||||||
|
@GetMapping("/md/delete/{fileName}")
|
||||||
|
@RequiresRoles(value = {RoleConstants.TEST_MANAGER, RoleConstants.TEST_USER,}, logical = Logical.OR)
|
||||||
|
public void delete(@PathVariable("fileName") String fileName) {
|
||||||
|
resourceService.mdDelete(fileName);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,10 @@
|
||||||
|
package io.metersphere.controller.request;
|
||||||
|
|
||||||
|
import lombok.Getter;
|
||||||
|
import lombok.Setter;
|
||||||
|
|
||||||
|
@Getter
|
||||||
|
@Setter
|
||||||
|
public class MdUploadRequest {
|
||||||
|
private String id;
|
||||||
|
}
|
|
@ -5,18 +5,19 @@ import io.metersphere.base.domain.JarConfig;
|
||||||
import io.metersphere.base.domain.JarConfigExample;
|
import io.metersphere.base.domain.JarConfigExample;
|
||||||
import io.metersphere.base.mapper.JarConfigMapper;
|
import io.metersphere.base.mapper.JarConfigMapper;
|
||||||
import io.metersphere.commons.exception.MSException;
|
import io.metersphere.commons.exception.MSException;
|
||||||
import io.metersphere.commons.utils.LogUtil;
|
import io.metersphere.commons.utils.FileUtils;
|
||||||
import io.metersphere.commons.utils.SessionUtils;
|
import io.metersphere.commons.utils.SessionUtils;
|
||||||
import io.metersphere.i18n.Translator;
|
import io.metersphere.i18n.Translator;
|
||||||
import org.apache.commons.lang3.StringUtils;
|
import org.apache.commons.lang3.StringUtils;
|
||||||
import org.aspectj.util.FileUtil;
|
|
||||||
import org.springframework.stereotype.Service;
|
import org.springframework.stereotype.Service;
|
||||||
import org.springframework.transaction.annotation.Transactional;
|
import org.springframework.transaction.annotation.Transactional;
|
||||||
import org.springframework.web.multipart.MultipartFile;
|
import org.springframework.web.multipart.MultipartFile;
|
||||||
|
|
||||||
import javax.annotation.Resource;
|
import javax.annotation.Resource;
|
||||||
import java.io.*;
|
import java.util.Collections;
|
||||||
import java.util.*;
|
import java.util.Comparator;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.UUID;
|
||||||
import java.util.stream.Collectors;
|
import java.util.stream.Collectors;
|
||||||
|
|
||||||
@Service
|
@Service
|
||||||
|
@ -65,7 +66,7 @@ public class JarConfigService {
|
||||||
|
|
||||||
public void delete(String id) {
|
public void delete(String id) {
|
||||||
JarConfig JarConfig = jarConfigMapper.selectByPrimaryKey(id);
|
JarConfig JarConfig = jarConfigMapper.selectByPrimaryKey(id);
|
||||||
deleteJarFile(JarConfig.getPath());
|
FileUtils.deleteFile(JarConfig.getPath());
|
||||||
jarConfigMapper.deleteByPrimaryKey(id);
|
jarConfigMapper.deleteByPrimaryKey(id);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -81,8 +82,8 @@ public class JarConfigService {
|
||||||
}
|
}
|
||||||
jarConfigMapper.updateByPrimaryKey(jarConfig);
|
jarConfigMapper.updateByPrimaryKey(jarConfig);
|
||||||
if (file != null) {
|
if (file != null) {
|
||||||
deleteJarFile(deletePath);
|
FileUtils.deleteFile(deletePath);
|
||||||
createJarFiles(file);
|
FileUtils.uploadFile(file, JAR_FILE_DIR);
|
||||||
NewDriverManager.loadJar(jarConfig.getPath());
|
NewDriverManager.loadJar(jarConfig.getPath());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -98,50 +99,15 @@ public class JarConfigService {
|
||||||
jarConfig.setPath(getJarPath(file));
|
jarConfig.setPath(getJarPath(file));
|
||||||
jarConfig.setFileName(file.getOriginalFilename());
|
jarConfig.setFileName(file.getOriginalFilename());
|
||||||
jarConfigMapper.insert(jarConfig);
|
jarConfigMapper.insert(jarConfig);
|
||||||
createJarFiles(file);
|
FileUtils.uploadFile(file, JAR_FILE_DIR);
|
||||||
NewDriverManager.loadJar(jarConfig.getPath());
|
NewDriverManager.loadJar(jarConfig.getPath());
|
||||||
return jarConfig.getId();
|
return jarConfig.getId();
|
||||||
}
|
}
|
||||||
|
|
||||||
public void deleteJarFiles(String testId) {
|
|
||||||
File file = new File(JAR_FILE_DIR + "/" + testId);
|
|
||||||
FileUtil.deleteContents(file);
|
|
||||||
if (file.exists()) {
|
|
||||||
file.delete();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void deleteJarFile(String path) {
|
|
||||||
File file = new File(path);
|
|
||||||
if (file.exists()) {
|
|
||||||
file.delete();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public String getJarPath(MultipartFile file) {
|
public String getJarPath(MultipartFile file) {
|
||||||
return JAR_FILE_DIR + "/" + file.getOriginalFilename();
|
return JAR_FILE_DIR + "/" + file.getOriginalFilename();
|
||||||
}
|
}
|
||||||
|
|
||||||
private String createJarFiles(MultipartFile jar) {
|
|
||||||
if (jar == null) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
File testDir = new File(JAR_FILE_DIR);
|
|
||||||
if (!testDir.exists()) {
|
|
||||||
testDir.mkdirs();
|
|
||||||
}
|
|
||||||
String filePath = testDir + "/" + jar.getOriginalFilename();
|
|
||||||
File file = new File(filePath);
|
|
||||||
try (InputStream in = jar.getInputStream(); OutputStream out = new FileOutputStream(file)) {
|
|
||||||
file.createNewFile();
|
|
||||||
FileUtil.copyStream(in, out);
|
|
||||||
} catch (IOException e) {
|
|
||||||
LogUtil.error(e.getMessage(), e);
|
|
||||||
MSException.throwException(Translator.get("upload_fail"));
|
|
||||||
}
|
|
||||||
return filePath;
|
|
||||||
}
|
|
||||||
|
|
||||||
private void checkExist(JarConfig jarConfig) {
|
private void checkExist(JarConfig jarConfig) {
|
||||||
if (jarConfig.getName() != null) {
|
if (jarConfig.getName() != null) {
|
||||||
JarConfigExample example = new JarConfigExample();
|
JarConfigExample example = new JarConfigExample();
|
||||||
|
|
|
@ -0,0 +1,46 @@
|
||||||
|
package io.metersphere.service;
|
||||||
|
|
||||||
|
import io.metersphere.commons.utils.FileUtils;
|
||||||
|
import io.metersphere.controller.request.MdUploadRequest;
|
||||||
|
import org.springframework.core.io.FileSystemResource;
|
||||||
|
import org.springframework.http.HttpHeaders;
|
||||||
|
import org.springframework.http.MediaType;
|
||||||
|
import org.springframework.http.ResponseEntity;
|
||||||
|
import org.springframework.stereotype.Service;
|
||||||
|
import org.springframework.transaction.annotation.Transactional;
|
||||||
|
import org.springframework.web.multipart.MultipartFile;
|
||||||
|
|
||||||
|
import java.io.File;
|
||||||
|
import java.util.Date;
|
||||||
|
|
||||||
|
@Service
|
||||||
|
@Transactional(rollbackFor = Exception.class)
|
||||||
|
public class ResourceService {
|
||||||
|
|
||||||
|
// private static final String RESOURCE_DIR = "/opt/metersphere/data/resource/";
|
||||||
|
private static final String MD_IMAGE_DIR = "/opt/metersphere/data/image/markdown";
|
||||||
|
|
||||||
|
public void mdUpload(MdUploadRequest request, MultipartFile file) {
|
||||||
|
FileUtils.uploadFile(file, MD_IMAGE_DIR, request.getId() + "_" + file.getOriginalFilename());
|
||||||
|
}
|
||||||
|
|
||||||
|
public ResponseEntity<FileSystemResource> getMdImage(String name) {
|
||||||
|
File file = new File(MD_IMAGE_DIR + "/" + name);
|
||||||
|
HttpHeaders headers = new HttpHeaders();
|
||||||
|
headers.add("Cache-Control", "no-cache, no-store, must-revalidate");
|
||||||
|
headers.add("Content-Disposition", "attachment; filename=" + file.getName());
|
||||||
|
headers.add("Pragma", "no-cache");
|
||||||
|
headers.add("Expires", "0");
|
||||||
|
headers.add("Last-Modified", new Date().toString());
|
||||||
|
headers.add("ETag", String.valueOf(System.currentTimeMillis()));
|
||||||
|
return ResponseEntity.ok()
|
||||||
|
.headers(headers)
|
||||||
|
.contentLength(file.length())
|
||||||
|
.contentType(MediaType.parseMediaType("application/octet-stream"))
|
||||||
|
.body(new FileSystemResource(file));
|
||||||
|
}
|
||||||
|
|
||||||
|
public void mdDelete(String fileName) {
|
||||||
|
FileUtils.deleteFile(MD_IMAGE_DIR + "/" + fileName);
|
||||||
|
}
|
||||||
|
}
|
|
@ -39,6 +39,7 @@
|
||||||
"lodash.isnumber": "^3.0.3",
|
"lodash.isnumber": "^3.0.3",
|
||||||
"lodash.isobject": "^3.0.2",
|
"lodash.isobject": "^3.0.2",
|
||||||
"lodash.isstring": "^4.0.1",
|
"lodash.isstring": "^4.0.1",
|
||||||
|
"mavon-editor": "^2.9.1",
|
||||||
"md5": "^2.3.0",
|
"md5": "^2.3.0",
|
||||||
"mockjs": "^1.1.0",
|
"mockjs": "^1.1.0",
|
||||||
"nprogress": "^0.2.0",
|
"nprogress": "^0.2.0",
|
||||||
|
|
|
@ -1,23 +1,84 @@
|
||||||
<template>
|
<template>
|
||||||
<el-form-item :disable="true" :label="title" :prop="prop" :label-width="labelWidth">
|
<el-form-item :disable="true" :label="title" :prop="prop" :label-width="labelWidth">
|
||||||
<test-case-rich-text :disabled="disabled" :content="data[prop]" @updateRichText="updateData"/>
|
<!-- <test-case-rich-text :disabled="disabled" :content="data[prop]" @updateRichText="updateData"/>-->
|
||||||
|
<mavon-editor :editable="!disabled" @imgAdd="imgAdd" :default-open="disabled ? 'preview' : null" class="mavon-editor"
|
||||||
|
:subfield="disabled ? false : true" :toolbars="toolbars" @imgDel="imgDel" v-model="data[prop]" ref="md"/>
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
import TestCaseRichText from "@/business/components/track/case/components/MsRichText";
|
import {getUUID} from "@/common/js/utils";
|
||||||
export default {
|
export default {
|
||||||
name: "FormRichTextItem",
|
name: "FormRichTextItem",
|
||||||
components: {TestCaseRichText},
|
components: {},
|
||||||
props: ['data', 'title', 'prop', 'disabled', 'labelWidth'],
|
props: ['data', 'title', 'prop', 'disabled', 'labelWidth'],
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
toolbars: {
|
||||||
|
bold: true, // 粗体
|
||||||
|
italic: true, // 斜体
|
||||||
|
header: true, // 标题
|
||||||
|
underline: true, // 下划线
|
||||||
|
strikethrough: true, // 中划线
|
||||||
|
mark: true, // 标记
|
||||||
|
superscript: true, // 上角标
|
||||||
|
subscript: true, // 下角标
|
||||||
|
quote: true, // 引用
|
||||||
|
ol: true, // 有序列表
|
||||||
|
ul: true, // 无序列表
|
||||||
|
link: true, // 链接
|
||||||
|
imagelink: true, // 图片链接
|
||||||
|
code: true, // code
|
||||||
|
table: true, // 表格
|
||||||
|
fullscreen: true, // 全屏编辑
|
||||||
|
readmodel: true, // 沉浸式阅读
|
||||||
|
htmlcode: true, // 展示html源码
|
||||||
|
help: true, // 帮助
|
||||||
|
/* 1.3.5 */
|
||||||
|
undo: true, // 上一步
|
||||||
|
redo: true, // 下一步
|
||||||
|
trash: true, // 清空
|
||||||
|
save: false, // 保存(触发events中的save事件)
|
||||||
|
/* 1.4.2 */
|
||||||
|
navigation: true, // 导航目录
|
||||||
|
/* 2.1.8 */
|
||||||
|
alignleft: true, // 左对齐
|
||||||
|
aligncenter: true, // 居中
|
||||||
|
alignright: true, // 右对齐
|
||||||
|
/* 2.2.1 */
|
||||||
|
subfield: true, // 单双栏模式
|
||||||
|
preview: true, // 预览
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
methods: {
|
methods: {
|
||||||
updateData(value) {
|
updateData(value) {
|
||||||
this.data[this.prop] = value;
|
this.data[this.prop] = value;
|
||||||
|
},
|
||||||
|
imgAdd(pos, file){
|
||||||
|
let param = {
|
||||||
|
id: getUUID().substring(0, 8)
|
||||||
|
};
|
||||||
|
file.prefix = param.id;
|
||||||
|
this.result = this.$fileUpload('/resource/md/upload', file, null, param, () => {
|
||||||
|
this.$success(this.$t('commons.save_success'));
|
||||||
|
this.$refs.md.$img2Url(pos, '/resource/md/get/' + param.id + '_' + file.name);
|
||||||
|
});
|
||||||
|
this.$emit('imgAdd', file);
|
||||||
|
},
|
||||||
|
imgDel(file) {
|
||||||
|
if (file) {
|
||||||
|
this.$get('/resource/md/delete/' + file[1].prefix + "_" + file[1].name);
|
||||||
}
|
}
|
||||||
|
},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style scoped>
|
<style scoped>
|
||||||
|
|
||||||
|
.mavon-editor {
|
||||||
|
min-height: 20px;
|
||||||
|
}
|
||||||
|
|
||||||
</style>
|
</style>
|
||||||
|
|
|
@ -79,8 +79,6 @@
|
||||||
|
|
||||||
<test-case-step-item :label-width="formLabelWidth" v-if="form.stepModel === 'STEP'" :form="form" :read-only="readOnly"/>
|
<test-case-step-item :label-width="formLabelWidth" v-if="form.stepModel === 'STEP'" :form="form" :read-only="readOnly"/>
|
||||||
|
|
||||||
<ms-form-divider :title="$t('test_track.case.other_info')"/>
|
|
||||||
|
|
||||||
<test-case-edit-other-info :project-id="projectIds" :form="form" :label-width="formLabelWidth" :case-id="form.id" ref="otherInfo"/>
|
<test-case-edit-other-info :project-id="projectIds" :form="form" :label-width="formLabelWidth" :case-id="form.id" ref="otherInfo"/>
|
||||||
|
|
||||||
<el-row style="margin-top: 10px" v-if="type!='add'">
|
<el-row style="margin-top: 10px" v-if="type!='add'">
|
||||||
|
@ -217,8 +215,8 @@ export default {
|
||||||
maintainer: [{required: true, message: this.$t('test_track.case.input_maintainer'), trigger: 'change'}],
|
maintainer: [{required: true, message: this.$t('test_track.case.input_maintainer'), trigger: 'change'}],
|
||||||
priority: [{required: true, message: this.$t('test_track.case.input_priority'), trigger: 'change'}],
|
priority: [{required: true, message: this.$t('test_track.case.input_priority'), trigger: 'change'}],
|
||||||
method: [{required: true, message: this.$t('test_track.case.input_method'), trigger: 'change'}],
|
method: [{required: true, message: this.$t('test_track.case.input_method'), trigger: 'change'}],
|
||||||
prerequisite: [{max: 500, message: this.$t('test_track.length_less_than') + '500', trigger: 'blur'}],
|
// prerequisite: [{max: 500, message: this.$t('test_track.length_less_than') + '500', trigger: 'blur'}],
|
||||||
remark: [{max: 1000, message: this.$t('test_track.length_less_than') + '1000', trigger: 'blur'}]
|
// remark: [{max: 1000, message: this.$t('test_track.length_less_than') + '1000', trigger: 'blur'}]
|
||||||
},
|
},
|
||||||
customFieldRules: {},
|
customFieldRules: {},
|
||||||
customFieldForm: {},
|
customFieldForm: {},
|
||||||
|
|
|
@ -2,7 +2,7 @@
|
||||||
<el-tabs class="other-info-tabs" v-loading="result.loading" v-model="tabActiveName">
|
<el-tabs class="other-info-tabs" v-loading="result.loading" v-model="tabActiveName">
|
||||||
<el-tab-pane :label="$t('commons.remark')" name="remark">
|
<el-tab-pane :label="$t('commons.remark')" name="remark">
|
||||||
<el-row>
|
<el-row>
|
||||||
<ms-rich-text :disabled="readOnly" :content="form.remark" @updateRichText="updateRemark"/>
|
<form-rich-text-item class="remark-item" :disabled="readOnly" :data="form" prop="remark"/>
|
||||||
</el-row>
|
</el-row>
|
||||||
</el-tab-pane>
|
</el-tab-pane>
|
||||||
<el-tab-pane :label="$t('test_track.case.relate_test')" name="relateTest">
|
<el-tab-pane :label="$t('test_track.case.relate_test')" name="relateTest">
|
||||||
|
@ -89,10 +89,11 @@ import {TEST} from "@/business/components/api/definition/model/JsonData";
|
||||||
import TestCaseAttachment from "@/business/components/track/case/components/TestCaseAttachment";
|
import TestCaseAttachment from "@/business/components/track/case/components/TestCaseAttachment";
|
||||||
import TestCaseIssueRelate from "@/business/components/track/case/components/TestCaseIssueRelate";
|
import TestCaseIssueRelate from "@/business/components/track/case/components/TestCaseIssueRelate";
|
||||||
import {enableModules} from "@/common/js/utils";
|
import {enableModules} from "@/common/js/utils";
|
||||||
|
import FormRichTextItem from "@/business/components/track/case/components/FormRichTextItem";
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
name: "TestCaseEditOtherInfo",
|
name: "TestCaseEditOtherInfo",
|
||||||
components: {TestCaseIssueRelate, TestCaseAttachment, MsRichText, TestCaseRichText},
|
components: {FormRichTextItem, TestCaseIssueRelate, TestCaseAttachment, MsRichText, TestCaseRichText},
|
||||||
props: ['form', 'labelWidth', 'caseId', 'readOnly', 'projectId', 'isTestPlan'],
|
props: ['form', 'labelWidth', 'caseId', 'readOnly', 'projectId', 'isTestPlan'],
|
||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
|
@ -314,4 +315,8 @@ export default {
|
||||||
.other-info-tabs {
|
.other-info-tabs {
|
||||||
padding: 10px 60px;
|
padding: 10px 60px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.remark-item {
|
||||||
|
padding: 0px 15px;
|
||||||
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|
|
@ -156,6 +156,11 @@ name: "TestCaseMinder",
|
||||||
step.result = result;
|
step.result = result;
|
||||||
}
|
}
|
||||||
steps.push(step);
|
steps.push(step);
|
||||||
|
|
||||||
|
if (data.stepModel === 'TEXT') {
|
||||||
|
testCase.stepDescription = step.desc;
|
||||||
|
testCase.expectedResult = step.result;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
if (childData.changed) isChange = true;
|
if (childData.changed) isChange = true;
|
||||||
})
|
})
|
||||||
|
|
|
@ -11,11 +11,13 @@ export function getTestCaseDataMap(testCase, isDisable, setParamCallback) {
|
||||||
}
|
}
|
||||||
|
|
||||||
export function parseCase(item, dataMap, isDisable, setParamCallback) {
|
export function parseCase(item, dataMap, isDisable, setParamCallback) {
|
||||||
|
|
||||||
if (item.steps) {
|
if (item.steps) {
|
||||||
item.steps = JSON.parse(item.steps);
|
item.steps = JSON.parse(item.steps);
|
||||||
} else {
|
} else {
|
||||||
item.steps = [];
|
item.steps = [];
|
||||||
}
|
}
|
||||||
|
|
||||||
// if (item.tags && item.tags.length > 0) {
|
// if (item.tags && item.tags.length > 0) {
|
||||||
// item.tags = JSON.parse(item.tags);
|
// item.tags = JSON.parse(item.tags);
|
||||||
// }
|
// }
|
||||||
|
@ -28,7 +30,8 @@ export function parseCase(item, dataMap, isDisable, setParamCallback) {
|
||||||
resource: [i18n.t('api_test.definition.request.case')],
|
resource: [i18n.t('api_test.definition.request.case')],
|
||||||
type: item.type,
|
type: item.type,
|
||||||
method: item.method,
|
method: item.method,
|
||||||
maintainer: item.maintainer
|
maintainer: item.maintainer,
|
||||||
|
stepModel: item.stepModel
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (setParamCallback) {
|
if (setParamCallback) {
|
||||||
|
@ -54,6 +57,13 @@ function parseChildren(nodeItem, item, isDisable) {
|
||||||
nodeItem.children = [];
|
nodeItem.children = [];
|
||||||
let children = [];
|
let children = [];
|
||||||
_parseChildren(children, item.prerequisite, i18n.t('test_track.case.prerequisite'), isDisable);
|
_parseChildren(children, item.prerequisite, i18n.t('test_track.case.prerequisite'), isDisable);
|
||||||
|
if (item.stepModel === 'TEXT') {
|
||||||
|
let descNode = _parseChildren(children, item.stepDescription, null, isDisable);
|
||||||
|
if (descNode) {
|
||||||
|
descNode.children = [];
|
||||||
|
_parseChildren(descNode.children, item.expectedResult, null, isDisable);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
if (item.steps) {
|
if (item.steps) {
|
||||||
item.steps.forEach((step) => {
|
item.steps.forEach((step) => {
|
||||||
let descNode = _parseChildren(children, step.desc, undefined, isDisable);
|
let descNode = _parseChildren(children, step.desc, undefined, isDisable);
|
||||||
|
@ -64,6 +74,7 @@ function parseChildren(nodeItem, item, isDisable) {
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
}
|
||||||
_parseChildren(children, item.remark, i18n.t('commons.remark'), isDisable);
|
_parseChildren(children, item.remark, i18n.t('commons.remark'), isDisable);
|
||||||
nodeItem.children = children;
|
nodeItem.children = children;
|
||||||
}
|
}
|
||||||
|
|
|
@ -23,6 +23,11 @@ import JsonSchemaEditor from './components/common/json-schema/schema/index';
|
||||||
import JSONPathPicker from 'vue-jsonpath-picker';
|
import JSONPathPicker from 'vue-jsonpath-picker';
|
||||||
import VueClipboard from 'vue-clipboard2'
|
import VueClipboard from 'vue-clipboard2'
|
||||||
import vueMinderEditor from 'vue-minder-editor-plus'
|
import vueMinderEditor from 'vue-minder-editor-plus'
|
||||||
|
|
||||||
|
import mavonEditor from 'mavon-editor'
|
||||||
|
import 'mavon-editor/dist/css/index.css'
|
||||||
|
Vue.use(mavonEditor)
|
||||||
|
|
||||||
Vue.use(vueMinderEditor)
|
Vue.use(vueMinderEditor)
|
||||||
|
|
||||||
Vue.use(JsonSchemaEditor);
|
Vue.use(JsonSchemaEditor);
|
||||||
|
|
Loading…
Reference in New Issue