|
|
<template>
|
|
|
<!-- 根据visible属性的值控制整个组件内容的显示与隐藏,当visible为true时显示 -->
|
|
|
<div v-show="visible">
|
|
|
<!-- 定义一个表单,绑定了dataForm数据模型,使用了dataRule验证规则,设置了表单尺寸为迷你型,标签宽度为80px -->
|
|
|
<el-form :model="dataForm" :rules="dataRule" ref="dataForm" size="mini" label-width="80px">
|
|
|
<!-- 第一行布局,使用el-row包裹 -->
|
|
|
<el-row>
|
|
|
<!-- 占12列,用于放置文章标题相关的表单元素 -->
|
|
|
<el-col :span="12">
|
|
|
<!-- 文章标题的表单项,设置了必填验证,对应dataForm中的title属性 -->
|
|
|
<el-form-item label="文章标题" prop="title" required>
|
|
|
<!-- 使用el-input组件实现输入框,双向绑定dataForm.title,设置了最大长度为1024,有占位提示文字 -->
|
|
|
<el-input v-model="dataForm.title" :maxlength="1024" placeholder="标题"></el-input>
|
|
|
</el-form-item>
|
|
|
</el-col>
|
|
|
<!-- 同样占12列,用于放置文章类型相关的表单元素 -->
|
|
|
<el-col :span="12">
|
|
|
<!-- 文章类型的表单项,设置了必填验证,对应dataForm中的type属性 -->
|
|
|
<el-form-item label="文章类型" prop="type" required>
|
|
|
<!-- 使用el-select组件实现下拉选择框,双向绑定dataForm.type,有占位提示文字 -->
|
|
|
<el-select v-model="dataForm.type" placeholder="选择文章类型">
|
|
|
<!-- 循环遍历ARTICLE_TYPES对象,生成下拉选项,选项的标签显示为name,值为key,允许用户创建新的选项 -->
|
|
|
<el-option v-for="(name,key) in ARTICLE_TYPES" :key="name" :label="name" :value="key" allow-create></el-option>
|
|
|
</el-select>
|
|
|
</el-form-item>
|
|
|
</el-col>
|
|
|
</el-row>
|
|
|
<!-- 第二行布局 -->
|
|
|
<el-row>
|
|
|
<!-- 占12列,用于放置一级目录相关的表单元素 -->
|
|
|
<el-col :span="12">
|
|
|
<!-- 一级目录的表单项,对应dataForm中的category属性 -->
|
|
|
<el-form-item label="一级目录" prop="category">
|
|
|
<!-- 使用el-input组件实现输入框,双向绑定dataForm.category,设置了最大长度为50,有占位提示文字 -->
|
|
|
<el-input :maxlength="50" v-model="dataForm.category" placeholder="一级目录"></el-input>
|
|
|
</el-form-item>
|
|
|
</el-col>
|
|
|
<!-- 占12列,用于放置二级分类相关的表单元素 -->
|
|
|
<el-col :span="12">
|
|
|
<!-- 二级分类的表单项,对应dataForm中的subCategory属性 -->
|
|
|
<el-form-item label="二级分类" prop="subCategory">
|
|
|
<!-- 使用el-input组件实现输入框,双向绑定dataForm.subCategory,设置了最大长度为50,有占位提示文字 -->
|
|
|
<el-input :maxlength="50" v-model="dataForm.subCategory" placeholder="二级目录"></el-input>
|
|
|
</el-form-item>
|
|
|
</el-col>
|
|
|
</el-row>
|
|
|
<!-- 指向外链的表单项,对应dataForm中的targetLink属性 -->
|
|
|
<el-form-item label="指向外链" prop="targetLink">
|
|
|
<!-- 使用el-input组件实现输入框,双向绑定dataForm.targetLink,有占位提示文字 -->
|
|
|
<el-input v-model="dataForm.targetLink" placeholder="指向外链"></el-input>
|
|
|
</el-form-item>
|
|
|
<!-- 摘要的表单项,对应dataForm中的summary属性 -->
|
|
|
<el-form-item label="摘要" prop="summary">
|
|
|
<!-- 使用el-input组件实现文本域输入框,双向绑定dataForm.summary,设置了行数为3,最大长度为512,显示字数限制提示 -->
|
|
|
<el-input v-model="dataForm.summary" placeholder="摘要" type="textarea" rows="3" maxlength="512" show-word-limit></el-input>
|
|
|
</el-form-item>
|
|
|
<!-- 标签的表单项,使用了自定义的tags-editor组件,双向绑定dataForm.tags -->
|
|
|
<el-form-item label="标签" prop="tags">
|
|
|
<tags-editor v-model="dataForm.tags"></tags-editor>
|
|
|
</el-form-item>
|
|
|
<!-- 封面图的表单项,对应dataForm中的image属性 -->
|
|
|
<el-form-item label="封面图" prop="image">
|
|
|
<!-- 使用el-input组件实现输入框,双向绑定dataForm.image,有占位提示文字 -->
|
|
|
<el-input v-model="dataForm.image" placeholder="图片链接">
|
|
|
<!-- 插入一个自定义的OssUploader组件,用于上传图片,上传成功后将返回值赋给dataForm.image -->
|
|
|
<OssUploader slot="append" @uploaded="dataForm.image=$event"></OssUploader>
|
|
|
</el-input>
|
|
|
</el-form-item>
|
|
|
<!-- 引入一个富文本编辑器组件,双向绑定dataForm.content -->
|
|
|
<tinymce-editor ref="editor" v-model="dataForm.content"></tinymce-editor>
|
|
|
</el-form>
|
|
|
<!-- 距离顶部有一定间距,靠右对齐的按钮区域 -->
|
|
|
<div class="margin-top text-right">
|
|
|
<!-- 取消按钮,点击时通过$emit触发父组件的hide事件 -->
|
|
|
<el-button @click="$emit('hide')">取消</el-button>
|
|
|
<!-- 确定按钮,类型为主要按钮,点击时调用dataFormSubmit方法进行表单提交 -->
|
|
|
<el-button type="primary" @click="dataFormSubmit()">确定</el-button>
|
|
|
</div>
|
|
|
</div>
|
|
|
</template>
|
|
|
|
|
|
<script>
|
|
|
// 引入Vuex的mapState辅助函数,用于将Vuex中的状态映射到组件的计算属性中
|
|
|
import { mapState } from 'vuex'
|
|
|
export default {
|
|
|
name: 'article-add-or-update',
|
|
|
components: {
|
|
|
// 异步加载TinymceEditor组件,应该是用于编辑文章内容的富文本编辑器组件
|
|
|
TinymceEditor: () => import("@/components/tinymce-editor"),
|
|
|
// 异步加载tags-editor组件,用于编辑文章标签
|
|
|
tagsEditor: () => import("@/components/tags-editor"),
|
|
|
// 异步加载OssUploader组件,用于上传图片(可能是上传到OSS存储服务)
|
|
|
OssUploader: () => import('../oss/oss-uploader')
|
|
|
},
|
|
|
props: {
|
|
|
visible: {
|
|
|
type: Boolean,
|
|
|
default: false
|
|
|
}
|
|
|
},
|
|
|
data() {
|
|
|
return {
|
|
|
dataForm: {
|
|
|
id: "",
|
|
|
type: '1',
|
|
|
title: "",
|
|
|
content: "",
|
|
|
category: "",
|
|
|
subCategory: "",
|
|
|
summary: "",
|
|
|
tags: "",
|
|
|
openCount: 0,
|
|
|
targetLink: location.origin + "/client/#/article/${articleId}",
|
|
|
image: ""
|
|
|
},
|
|
|
dataRule: {
|
|
|
type: [
|
|
|
{ required: true, message: "文章类型不能为空", trigger: "blur" }
|
|
|
],
|
|
|
title: [
|
|
|
{ required: true, message: "标题不能为空", trigger: "blur" }
|
|
|
],
|
|
|
category: [
|
|
|
{ required: true, message: "分类不能为空", trigger: "blur" }
|
|
|
]
|
|
|
}
|
|
|
};
|
|
|
},
|
|
|
computed: mapState({
|
|
|
// 从Vuex的state中获取article模块下的ARTICLE_TYPES状态,映射到组件的计算属性中,用于文章类型下拉框的选项数据
|
|
|
ARTICLE_TYPES: state => state.article.ARTICLE_TYPES
|
|
|
}),
|
|
|
methods: {
|
|
|
init(id) {
|
|
|
// 初始化时设置dataForm的id属性,传入的id为空字符串则设置为空,否则使用传入的值
|
|
|
this.dataForm.id = id || "";
|
|
|
this.$nextTick(() => {
|
|
|
// 表单DOM更新后,重置表单所有字段的值
|
|
|
this.$refs["dataForm"].resetFields();
|
|
|
if (id > 0) {
|
|
|
// 如果传入的id大于0,表示是编辑已有文章,发起HTTP请求获取文章详情信息
|
|
|
this.$http({
|
|
|
url: this.$http.adornUrl(`/manage/article/info/${this.dataForm.id}`),
|
|
|
method: "get",
|
|
|
params: this.$http.adornParams()
|
|
|
}).then(({ data }) => {
|
|
|
if (data && data.code === 200) {
|
|
|
// 如果请求成功,将返回的文章数据赋值给dataForm,同时将文章类型转为字符串类型(可能原数据类型不一致)
|
|
|
this.dataForm = data.article;
|
|
|
this.dataForm.type = data.article.type + "";
|
|
|
}
|
|
|
});
|
|
|
}
|
|
|
});
|
|
|
},
|
|
|
// 表单提交方法
|
|
|
dataFormSubmit() {
|
|
|
// 对表单进行验证,验证结果通过回调函数的参数valid返回
|
|
|
this.$refs["dataForm"].validate(valid => {
|
|
|
if (valid) {
|
|
|
// 如果表单验证通过,发起HTTP请求保存文章数据(可能是新增或更新操作,根据后端接口逻辑)
|
|
|
this.$http({
|
|
|
url: this.$http.adornUrl(`/manage/article/save`),
|
|
|
method: "post",
|
|
|
data: this.$http.adornData(this.dataForm)
|
|
|
}).then(({ data }) => {
|
|
|
if (data && data.code === 200) {
|
|
|
// 如果保存成功,弹出成功提示信息,提示框关闭后通过$emit触发父组件的refreshDataList和hide事件
|
|
|
this.$message({
|
|
|
message: "操作成功",
|
|
|
type: "success",
|
|
|
duration: 1500,
|
|
|
onClose: () => {
|
|
|
this.$emit("refreshDataList");
|
|
|
this.$emit('hide')
|
|
|
}
|
|
|
});
|
|
|
} else {
|
|
|
// 如果保存失败,弹出错误提示信息
|
|
|
this.$message.error(data.msg);
|
|
|
}
|
|
|
});
|
|
|
}
|
|
|
});
|
|
|
},
|
|
|
imgUploadSuccess(response, file, fileList) {
|
|
|
console.log(response);
|
|
|
if (response.code == 200) {
|
|
|
// 如果图片上传成功(根据返回数据的状态码判断),将返回的图片数据赋值给dataForm.image
|
|
|
this.dataForm.image = response.data;
|
|
|
console.log("this.article", this.article);
|
|
|
} else {
|
|
|
// 如果上传失败,弹出警告提示信息
|
|
|
this.$message.warning(response.msg);
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
};
|
|
|
</script> |