粒子研究院后台前端
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

539 lines
19 KiB

2 years ago
<template>
<div>
<!--
2 years ago
模块type: introduce | form | forms | column | columns | history
-->
10 months ago
<el-dialog title="模块管理" :visible.sync="visible" :width="dialogWidth" custom-class="module"
:close-on-click-modal="false" :before-close="close">
2 years ago
<template v-if="data.type === 'introduce'">
10 months ago
<el-table class="module-table" :data="data.list" header-align="center" row-key="id">
<el-table-column type="index" width="60" label="序号" align="center"></el-table-column>
2 years ago
10 months ago
<el-table-column v-for="(item, i) in data.forms" :key="i" :prop="item.prop" :label="item.label"
min-width="130" align="center">
2 years ago
<template slot-scope="scope">
10 months ago
<div v-if="item.type === 'link' && scope.row.link" class="link-wrap">
2 years ago
<span>{{ scope.row.link.linkName }}</span>
</div>
2 years ago
<template v-else-if="item.type === 'upload'">
10 months ago
<img v-if="scope.row.pic" :src="scope.row.pic" class="upload-pic">
<div class="upload-none" v-else>
2 years ago
<i class="el-icon-picture-outline"></i>
</div>
</template>
<p v-else>{{ scope.row[item.prop] }}</p>
2 years ago
</template>
</el-table-column>
2 years ago
10 months ago
<el-table-column label="操作" :width="data.sort ? 150 : 100" align="center">
2 years ago
<template slot-scope="scope">
<div class="flex a-center">
10 months ago
<el-switch v-model="scope.row.isEnable" :active-value="1" :inactive-value="0">
2 years ago
</el-switch>
10 months ago
<i class="el-icon-edit-outline del" @click="editIntro(scope.row, scope.$index)"></i>
2 years ago
<template v-if="data.sort">
10 months ago
<i v-if="scope.$index != data.list.length - 1" class="el-icon-bottom del"
@click="sort(1, scope.$index)"></i>
<i v-if="scope.$index" class="el-icon-top del" @click="sort(0, scope.$index)"></i>
2 years ago
</template>
10 months ago
<i class="el-icon-delete del" @click="delRow(data.list, scope.$index)"></i>
2 years ago
</div>
</template>
</el-table-column>
</el-table>
<div class="plus">
10 months ago
<i class="el-icon-circle-plus-outline" @click="addRow"></i>
2 years ago
</div>
</template>
10 months ago
<el-form v-else-if="data.type === 'form' || data.type === 'forms'" ref="form" :model="data.form" :rules="rules"
label-width="70px">
<el-form-item v-for="(item, i) in data.forms" :key="i" :prop="item.prop" :label="item.label">
<el-input v-if="item.type === 'input'" v-model="data.form[item.prop]" placeholder="请输入"
maxlength="100"></el-input>
<el-input v-if="item.type === 'textarea'" v-model="data.form[item.prop]" type="textarea" placeholder="请输入"
maxlength="300"></el-input>
<div v-if="item.type === 'upload' || item.type === 'video' || item.type === 'media'" class="uploader-wrap">
<el-upload v-if="item.type === 'upload'" class="uploader" accept=".jpg,.png,.jpeg,.gif"
:on-change="res => changeFile(res, data.form)" :show-file-list="false" :headers="headers"
:action="api.upload">
<img v-if="data.form.pic && !isVideo(data.form.mediaType)" :src="data.form.pic" class="avatar">
<div class="uploader-default" v-else>
2 years ago
<i class="el-icon-plus"></i>
<p>上传图片</p>
</div>
10 months ago
<div slot="tip" class="el-upload__tip">
1 year ago
<p>只支持.jpg,.png格式</p>
2 years ago
</div>
</el-upload>
10 months ago
<el-upload v-if="item.type === 'video'" accept=".mp4,.mov,.avi"
:on-success="res => uploadSuccess(res, data.form, item)" :before-remove="beforeRemove"
:on-remove="(file, fileList) => handleRemove(file, fileList, data.form, item.prop)" :file-list="fileList"
:headers="headers" :action="api.upload">
1 year ago
2 years ago
<el-button>上传视频</el-button>
10 months ago
<div slot="tip" class="el-upload__tip">
1 year ago
<p>请上传大小1G以内的视频</p>
</div>
</el-upload>
<!-- 图片视频都可上传 -->
10 months ago
<el-upload v-if="item.type === 'media'" :on-success="res => uploadSuccess(res, data.form, item)"
:before-remove="beforeRemove"
:on-remove="(file, fileList) => handleRemove(file, fileList, data.form, item.prop)" :file-list="fileList"
:headers="headers" :action="api.upload">
2 years ago
<el-button>上传资源</el-button>
</el-upload>
10 months ago
<i v-if="!item.required" class="el-icon-delete del" @click="data.form[item.prop] = ''"></i>
</div>
10 months ago
<div v-if="item.type === 'link'" class="flex">
<el-input class="m-r-10" v-model="data.form.link.linkName"></el-input>
2 years ago
<el-button @click="toLink(data.form)">设置链接</el-button>
</div>
10 months ago
<Editor v-if="item.type === 'editor'" api-key='rnk6zw9v267xqz7pf98twt1vmrvltmd436je7a642pckltda'
v-model="data.form[item.prop]" :init="editorConfig" />
2 years ago
</el-form-item>
</el-form>
10 months ago
<el-form v-else-if="data.type === 'column' || data.type === 'columns'" ref="form" :model="data.form"
:rules="columnRules" label-width="100px">
<el-form-item prop="site" label="站点选择">
<el-select v-model="data.form.site" @change="siteChange">
<el-option v-for="item in $refs.link.sites" :key="item.id" :label="item.siteName" :value="item.id">
2 years ago
</el-option>
</el-select>
</el-form-item>
10 months ago
<el-form-item prop="column" label="关联栏目">
<el-cascader ref="links" v-model="data.form.column" :options="columns" :props="columnProps" clearable
@change="columnChange"></el-cascader>
2 years ago
</el-form-item>
<el-form-item label="文章展示数量">
<el-select v-model="data.form.articleNum">
10 months ago
<el-option v-for="item in articleNums" :key="item.id" :label="item.name" :value="item.id">
2 years ago
</el-option>
</el-select>
</el-form-item>
<el-form-item label="栏目标题">
10 months ago
<el-radio v-model="data.form.columnTitle" :label="1" @change="columnTitleChange">默认读取关联栏目</el-radio>
2 years ago
<div class="radio-wrap">
10 months ago
<el-radio v-model="data.form.columnTitle" :label="2" @change="columnTitleChange"></el-radio>
<el-input v-model="data.form.columnTitleCustom" :disabled="data.form.columnTitle == 1" size="small"
maxlength="100"></el-input>
2 years ago
</div>
</el-form-item>
</el-form>
<template v-else-if="data.type === 'history'">
10 months ago
<el-table class="module-table" :data="data.list" header-align="center" row-key="id">
<el-table-column type="index" width="60" label="序号" align="center"></el-table-column>
<el-table-column prop="title" label="标题" min-width="140" align="center">
<template slot-scope="scope">
10 months ago
<el-input v-model="scope.row.title" placeholder="请输入" maxlength="100"></el-input>
</template>
</el-table-column>
10 months ago
<el-table-column label="操作" width="100" align="center">
<template slot-scope="scope">
<div class="flex a-center">
10 months ago
<el-switch v-model="scope.row.isEnable" :active-value="1" :inactive-value="0">
</el-switch>
10 months ago
<i class="el-icon-edit-outline del" @click="editHistory(scope.row, scope.$index)"></i>
<i class="el-icon-delete del" @click="delRow(data.list, scope.$index)"></i>
</div>
</template>
</el-table-column>
</el-table>
<div class="plus">
10 months ago
<i class="el-icon-circle-plus-outline" @click="addRow"></i>
</div>
</template>
10 months ago
<span slot="footer" class="dialog-footer">
2 years ago
<el-button @click="$emit('update:visible', false)">取消</el-button>
10 months ago
<el-button type="primary" @click="moduleSubmit">确定</el-button>
2 years ago
</span>
</el-dialog>
10 months ago
<Link ref="link" :data.sync="linkForm" :visible.sync="linkVisible" @linkSubmit="linkSubmit" />
<Content :data.sync="data" :visible.sync="contentVisible" @contentSubmit="contentSubmit" />
<History :data.sync="data" :list.sync="historyData" :form.sync="data.form" :visible.sync="historyVisible"
@historySubmit="historySubmit" />
2 years ago
<!-- 剪裁组件弹窗 -->
10 months ago
<el-dialog title="图片裁剪" append-to-body :visible.sync="cropperModel" width="1100px" :close-on-click-modal="false">
<Cropper ref="cropper" :img-file.sync="file" :is-upload="isUpload" :fixed="fixed" :fixedNumber.sync="fixedNumber"
:autoCropWidth="autoCropWidth" :autoCropHeight="autoCropHeight" :dataType="data.type" @upload="customUpload" />
2 years ago
</el-dialog>
2 years ago
</div>
</template>
<script>
import Link from '@/components/modules/link'
2 years ago
import Content from '@/components/modules/content'
import History from '@/components/modules/history'
import Editor from '@tinymce/tinymce-vue'
2 years ago
import Setting from '@/setting'
import Util from '@/libs/util'
2 years ago
import 'tinymce-paragraphspacing'
import editorConfig from '@/components/editor'
2 years ago
import Cropper from '@/components/img-upload/Cropper'
import Axios from 'axios'
2 years ago
import { mapActions } from 'vuex'
2 years ago
export default {
2 years ago
name: 'module',
2 years ago
props: ['data', 'visible'],
components: {
2 years ago
Link,
Content,
Editor,
2 years ago
History,
Cropper
2 years ago
},
2 years ago
data () {
2 years ago
return {
2 years ago
site: this.$store.state.content.site,
2 years ago
headers: {
token: Util.local.get(Setting.tokenKey)
},
2 years ago
isVideo: Util.isVideo,
2 years ago
dialogWidth: '800px',
2 years ago
rules: {},
linkVisible: false,
linkForm: {},
2 years ago
curIndex: 0,
editorConfig,
2 years ago
columnRules: {
site: [
{ required: true, message: '请选择站点', trigger: 'change' }
],
column: [
{ required: true, message: '请选择关联栏目', trigger: 'change' }
],
},
columns: [],
articleNums: [
{
id: '',
name: '默认'
},
{
id: 4,
name: '4'
},
{
id: 3,
name: '3'
},
{
id: 2,
name: '2'
},
],
2 years ago
fileList: [],
columnProps: {
checkStrictly: true,
value: 'id',
label: 'columnName'
},
2 years ago
contentVisible: false,
curModule: 0,
curData: {},
historyVisible: false,
2 years ago
historyData: [],
cropperModel: false,
isUpload: false,
file: {}, // 当前被选择的图片文件
fixed: false,
fixedNumber: [0.88, 1],
autoCropWidth: 480,
autoCropHeight: 124,
2 years ago
};
},
watch: {
2 years ago
visible (open) {
2 years ago
// 每次打开的时候处理参数
open && this.handleForm()
}
},
2 years ago
mounted () {
2 years ago
this.handleForm()
},
methods: {
2 years ago
...mapActions("user", [
'logout'
]),
2 years ago
// 处理form表单参数
2 years ago
handleForm () {
2 years ago
const { type, forms, form, dialogWidth } = this.data
this.fixed = false
2 years ago
if (type === 'form' || type === 'forms') {
2 years ago
forms.map(e => {
2 years ago
if (e.type === 'video' || e.type === 'media') {
this.fileList = [
{
name: form.fileName,
url: form.video || e.pic
}
]
}
2 years ago
if (e.type === 'upload' && e.width) {
this.autoCropWidth = e.width
this.autoCropHeight = e.height
this.fixed = true
this.fixedNumber = [e.width / e.height, 1]
}
2 years ago
// 加必填
2 years ago
if (e.required) {
this.rules[e.prop] = [
{
required: true,
message: `${e.type === 'input' ? '输入' : '选择'}${e.label}`,
trigger: e.type === 'input' ? 'blur' : 'change'
}
]
}
})
}
2 years ago
if (type === 'column' && form.articleNumOpt) {
const nums = [
{
id: '',
name: '默认'
}
]
for (let i = form.articleNumOpt; i > 1; i--) {
nums.push({
id: i,
name: i
})
}
this.articleNums = nums
}
2 years ago
form && form.site && this.siteChange(form.site)
2 years ago
this.dialogWidth = dialogWidth || '800px'
},
// 站点切换回调
2 years ago
siteChange (siteId) {
1 year ago
this.$post(this.api.listWithTree, Util.rsa({
siteId,
columnName: '',
templateId: '',
2 years ago
typeId: '',
isSort: 1
1 year ago
})).then(({ data }) => {
this.data.type === 'columns' || this.handleColumn(data)
this.columns = data
2 years ago
}).catch(err => { })
},
// 递归处理栏目
2 years ago
handleColumn (data) {
data.map(e => {
// if (e.typeId === 2 || e.typeId === 3) e.disabled = true
e.children.length ? this.handleColumn(e.children) : (delete e.children)
})
},
2 years ago
// 栏目切换回调
2 years ago
columnChange (val) {
2 years ago
this.data.form.columnName = this.$refs.links.getCheckedNodes()[0].label
},
2 years ago
// 栏目标题切换回调
2 years ago
columnTitleChange (val) {
2 years ago
const { form } = this.data
const checked = this.$refs.links.getCheckedNodes()
form.columnName = val === 2 ? form.columnTitleCustom : (checked.length ? checked[0].label : '')
},
// 添加行
2 years ago
addRow () {
const { type } = this.data
2 years ago
if (type === 'history') {
this.data.list.push(JSON.parse(JSON.stringify(this.data.historyForm)))
2 years ago
} else {
this.editIntro(this.data.originForm, -1)
}
2 years ago
},
2 years ago
close () {
2 years ago
this.$emit('update:visible', false)
},
2 years ago
// 图片裁剪上传事件
customUpload (data) {
const formData = new FormData()
formData.append('file', data, this.file.name)
this.imgUpload(formData)
},
// 图片上传到服务器
imgUpload (formData) {
this.isUpload = true
Axios({
method: 'post',
url: this.api.upload,
data: formData,
headers: {
'Content-Type': 'multipart/form-data',
token: Util.local.get(Setting.tokenKey)
},
}).then(({ data }) => {
1 year ago
if (data.code === 401) {
Util.errorMsg("登录过期,请重新登录");
setTimeout(() => {
this.$store.dispatch('user/logout')
}, 1000);
return false
}
2 years ago
this.$set(this.curForm, 'pic', data.url)
}).catch(res => { })
this.$refs.cropper.isDisabled = false
this.isUpload = false
this.cropperModel = false
},
// 图片改变钩子
changeFile (file, form) {
const { size, name } = file
const ext = name.substring(name.lastIndexOf('.') + 1)
if (!Util.isImg(ext)) {
this.$message.error('请上传图片!')
return false
}
1 year ago
if (size / 1024 / 1024 > 100) {
this.$message.error('请上传100M以内的图片!')
2 years ago
return false
}
this.file = file
this.curForm = form
this.cropperModel = true
this.$nextTick(() => {
this.$refs.cropper.updateImg({
url: window.URL.createObjectURL(file.raw),
size: file.size
})
})
},
2 years ago
// 上传成功
2 years ago
uploadSuccess (res, row, item = {}) {
if (res.code === 401) {
Util.errorMsg("登录过期,请重新登录");
setTimeout(() => {
this.logout()
}, 1000)
} else {
this.$set(row, item.type === 'video' ? 'video' : 'pic', res.url)
this.$set(row, 'mediaType', res.original.substr(res.original.lastIndexOf('.') + 1))
this.$set(row, 'fileName', res.original)
}
},
beforeRemove (file, fileList) {
return this.$confirm(`确定移除 ${file.name}`);
},
handleRemove (file, fileList, form, prop) {
form[prop] = ''
form.fileName = ''
2 years ago
},
2 years ago
// 排序
sort (type, i) {
const item = this.data.list.splice(i, 1)[0]
this.data.list.splice(i + (type ? 1 : -1), 0, item)
},
2 years ago
// 删除行(通用)
2 years ago
delRow (list, i) {
2 years ago
this.$confirm('确定要删除吗?', '提示', {
type: 'warning'
}).then(() => {
list.splice(i, 1)
2 years ago
}).catch(() => { })
2 years ago
},
// 展示链接设置
2 years ago
toLink (row, i = 0) {
2 years ago
if (!row.link.linkName) {
row.link = {
linkName: '无',
connectionType: 1,
columnId: [],
articleId: '',
2 years ago
linkAddress: '',
2 years ago
site: this.site.id,
otherColumnId: [],
otherArticleId: '',
isOpen: 1
}
}
2 years ago
this.curIndex = i
this.linkForm = row.link
this.linkVisible = true
2 years ago
},
2 years ago
// 编辑内容
2 years ago
editIntro (row, i = 0) {
this.data.form = JSON.parse(JSON.stringify(row))
2 years ago
if (!this.data.labelWidth) this.data.labelWidth = '60px'
this.curModule = i
2 years ago
this.contentVisible = true
},
// content设置提交
2 years ago
contentSubmit () {
this.contentVisible = false
const { form } = this.data
2 years ago
if (this.curModule === -1) {
this.data.list.push(JSON.parse(JSON.stringify(this.data.form)))
} else {
const list = this.data.list[this.curModule]
for (const i in form) {
this.$set(list, i, form[i])
}
}
this.data.form = JSON.parse(JSON.stringify(this.data.originForm))
},
2 years ago
// 链接设置提交
2 years ago
linkSubmit () {
2 years ago
const el = this.$refs.link
const data = this.data.form ? this.data.form.link : this.data.list[this.curIndex].link
2 years ago
let name
// 站内链接
if (data.connectionType === 1) {
if (!data.columnId.length) return Util.errorMsg('请选择站内链接')
// 如果选择了文章,则取文章标题作为链接名称,否则取栏目标题
if (data.articleId) {
const item = el.articles.find(e => e.id == data.articleId)
name = item ? item.title : ''
} else {
name = el.$refs.column.getCheckedNodes()[0].pathLabels.join('/')
2 years ago
}
2 years ago
} else if (data.connectionType === 3) { // 其他站点链接
if (!data.otherColumnId.length) return Util.errorMsg('请选择栏目')
// 如果选择了文章,则取文章标题作为链接名称,否则取栏目标题
if (data.otherArticleId) {
const item = el.articles.find(e => e.id == data.otherArticleId)
name = item ? item.title : ''
} else {
name = el.$refs.otherColumn.getCheckedNodes()[0].pathLabels.join('/')
2 years ago
}
2 years ago
} else { // 站外链接
if (!data.linkAddress) return Util.errorMsg('请输入站外链接')
name = data.linkAddress
2 years ago
}
2 years ago
data.linkName = name
2 years ago
this.linkVisible = false
},
// 编辑history
2 years ago
editHistory (row, i = 0) {
this.historyData = row.list
this.curModule = i
this.historyVisible = true
},
// history设置提交
2 years ago
historySubmit () {
this.historyVisible = false
2 years ago
},
// 模块设置提交
2 years ago
moduleSubmit () {
this.$emit('moduleSubmit')
2 years ago
},
2 years ago
}
};
</script>
<style lang="scss" scoped>
2 years ago
.radio-wrap {
10 months ago
display: flex;
align-items: center;
.el-input {
width: 200px;
margin-left: -40px;
}
2 years ago
}
2 years ago
</style>