粒子研究院后台前端
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.

645 lines
22 KiB

2 years ago
<template>
<div>
<!--
2 years ago
模块type: banner | introduce | form | forms | column | columns | history | customList
-->
2 years ago
<el-dialog title="模块管理"
:visible.sync="visible"
width="700px"
custom-class="module"
:close-on-click-modal="false"
:before-close="close">
2 years ago
<template v-if="data.type === 'banner' || data.type === 'introduce' || data.type === 'label' || data.type === 'customList'">
2 years ago
<el-table v-if="data.type === 'label'"
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">
2 years ago
<el-input v-model="scope.row.title"
placeholder="请输入"
maxlength="100"></el-input>
</template>
</el-table-column>
2 years ago
<el-table-column label="操作"
width="60"
align="center">
<template slot-scope="scope">
<div class="flex j-center a-center">
2 years ago
<i class="el-icon-delete del"
@click="delRow(data.list, scope.$index)"></i>
</div>
</template>
</el-table-column>
</el-table>
2 years ago
<el-table v-else-if="data.type === 'customList'"
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 v-for="(item, i) in data.forms"
:key="i"
:prop="item.prop"
:label="item.label"
min-width="120"
align="center">
2 years ago
<template slot-scope="scope">
2 years ago
<el-input v-model="scope.row[item.prop]"
:placeholder="'请输入' + item.label"
maxlength="100"></el-input>
2 years ago
</template>
</el-table-column>
2 years ago
<el-table-column label="操作"
width="60"
align="center">
2 years ago
<template slot-scope="scope">
<div class="flex j-center a-center">
2 years ago
<i class="el-icon-delete del"
@click="delRow(data.list, scope.$index)"></i>
2 years ago
</div>
</template>
</el-table-column>
</el-table>
2 years ago
<el-table v-else
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
<el-table-column v-if="!data.notPic"
label="图片"
2 years ago
min-width="140"
align="center">
2 years ago
<template slot-scope="scope">
2 years ago
<el-upload v-if="data.type === 'banner'"
class="uploader"
accept=".jpg,.png,.jpeg"
:on-success="res => uploadSuccess(res, scope.row)"
:show-file-list="false"
:headers="headers"
:action="api.upload">
<img v-if="scope.row.pic"
:src="scope.row.pic"
class="avatar">
<div class="uploader-default"
v-else>
<i class="el-icon-plus"></i>
<p>上传图片</p>
</div>
<div slot="tip"
class="el-upload__tip">
<p>只支持.jpg,.png格式</p>
</div>
2 years ago
</el-upload>
<template v-else>
2 years ago
<img v-if="scope.row.pic"
:src="scope.row.pic"
class="upload-pic">
<div class="upload-none"
v-else>
<i class="el-icon-picture-outline"></i>
</div>
</template>
2 years ago
</template>
</el-table-column>
2 years ago
<el-table-column v-if="data.type === 'introduce'"
prop="title"
label="标题"
min-width="140"
align="center"></el-table-column>
<el-table-column label="链接"
min-width="140"
align="center">
2 years ago
<template slot-scope="scope">
<div class="link-wrap">
<span>{{ scope.row.link.linkName }}</span>
2 years ago
<el-button v-if="data.type === 'banner'"
class="set-link"
type="primary"
size="mini"
@click="toLink(scope.row, scope.$index)">设置链接</el-button>
2 years ago
</div>
</template>
</el-table-column>
2 years ago
<el-table-column label="操作"
width="100"
align="center">
2 years ago
<template slot-scope="scope">
<div class="flex a-center">
2 years ago
<el-switch v-model="scope.row.isEnable"
:active-value="1"
:inactive-value="0">
2 years ago
</el-switch>
2 years ago
<i v-if="data.type === 'introduce'"
class="el-icon-edit-outline del"
@click="editIntro(scope.row, scope.$index)"></i>
<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">
2 years ago
<i class="el-icon-circle-plus-outline"
@click="addRow"></i>
2 years ago
</div>
</template>
2 years ago
<el-form v-else-if="data.type === 'form' || data.type === 'forms'"
ref="form"
:model="data.form"
:rules="rules"
label-width="60px">
<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 class="uploader"
accept=".jpg,.png,.jpeg"
:on-success="res => uploadSuccess(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>
<i class="el-icon-plus"></i>
<p>上传图片</p>
</div>
<div slot="tip"
class="el-upload__tip">
<p>只支持.jpg,.png,.mp4格式</p>
</div>
</el-upload>
2 years 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">
<el-button>上传视频</el-button>
</el-upload>
<!-- 图片视频都可上传 -->
2 years 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">
<el-button>上传资源</el-button>
</el-upload>
2 years ago
<i v-if="!item.required"
class="el-icon-delete del"
@click="data.form[item.prop] = ''"></i>
</div>
2 years 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>
2 years 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>
2 years 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>
2 years 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">
2 years 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="栏目标题">
2 years ago
<el-radio v-model="data.form.columnTitle"
:label="1"
@change="columnTitleChange">默认读取关联栏目</el-radio>
2 years ago
<div class="radio-wrap">
2 years 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="10"></el-input>
2 years ago
</div>
</el-form-item>
</el-form>
<template v-else-if="data.type === 'history'">
2 years 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">
2 years ago
<el-input v-model="scope.row.title"
placeholder="请输入"
maxlength="100"></el-input>
</template>
</el-table-column>
2 years ago
<el-table-column label="操作"
width="100"
align="center">
<template slot-scope="scope">
<div class="flex a-center">
2 years ago
<el-switch v-model="scope.row.isEnable"
:active-value="1"
:inactive-value="0">
</el-switch>
2 years 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">
2 years ago
<i class="el-icon-circle-plus-outline"
@click="addRow"></i>
</div>
</template>
2 years ago
<span slot="footer"
class="dialog-footer">
2 years ago
<el-button @click="$emit('update:visible', false)">取消</el-button>
2 years ago
<el-button type="primary"
@click="moduleSubmit">确定</el-button>
2 years ago
</span>
</el-dialog>
2 years 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
</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'
import editorConfig from '@/components/editor'
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,
History
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
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,
historyData: []
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 } = this.data
2 years ago
if (type === 'form') {
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
// 加必填
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
siteChange (siteId) {
this.$post(this.api.listWithTree, {
siteId,
columnName: '',
templateId: '',
2 years ago
typeId: '',
isSort: 1
}).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 === 'banner' || type === 'label' || type === 'customList') {
2 years ago
this.data.list.push(JSON.parse(JSON.stringify(this.data.originForm)))
} else 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
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
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))
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 {
2 years ago
display: flex;
align-items: center;
.el-input {
width: 200px;
margin-left: -40px;
}
2 years ago
}
2 years ago
</style>