| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507 |
- <script>
- import axios from "axios";
- import {defineComponent} from "vue";
- import fieldIsAllow from "../../../until/fieldIsAllow"
- import RoundedTitle from "../../../components/public/roundedTitle.vue";
- import {rCode} from "../../../map/rcodeMap_esm";
- import handle from "../../../until/handle";
- import ImageViewer from "../../../components/public/imageViewer.vue";
- import ImageTable from "../../../components/public/imageTable.vue";
- import Pop from "../../../components/public/pop.vue";
- import PopCard from "../../../components/public/popCard.vue";
- import InputRow from "../../../components/public/form/inputRow.vue";
- import dbField_esm from "../../../map/dbField_esm";
- import {apiMap} from "../../../map/apiMap";
- import SearchBox from "../../../components/search/searchBox.vue";
- import {pTypes} from "../../../map/productMap";
- import {newsType} from "../../../map/newMap";
- export default defineComponent({
- name: 'carousel',
- computed: {
- dbField_esm() {
- return dbField_esm
- }
- },
- components: {SearchBox, InputRow, PopCard, Pop, ImageTable, ImageViewer, RoundedTitle},
- async asyncData(ctx){
- // 加载轮播图数据
- let [err,res] = await handle(axios.get(apiMap.carouselList.path));
- if(err){
- return {};
- }
- let result = res.data;
- if(result.code === rCode.OK){
- return {carouselList: result.data}
- }else{
- this.$message.error(result.msg);
- return {}
- }
- return {}
- },
- data(){
- return {
- limit: 10,
- loading: false,
- carouselList: [],
- popShow: false,
- popLoading: false,
- carouselPopShow: true,
- carouselPopLoading: false,
- isEditCarousel: false,
- carouselData: {},
- form: {
- // 排序
- sort: {
- val:0,
- init: 0,
- msg: '',
- state: 0,
- },
- // 状态 0:禁用,1:启用
- status: {
- val: 0,
- init: 0,
- msg: '',
- state: 0,
- },
- // 图像资源id, 0为无图
- file: {
- val: 0,
- init: 0,
- msg: '',
- state: 0,
- },
- // 轮播类型 '0:product','1:news','2:page','3:href'
- type: {
- val: 0,
- init: 0,
- msg: '',
- state: 0,
- options: [
- {label: '直接链接', value: 0},
- {label: '内部产品', value: 1},
- {label: '指向新闻', value: 2},
- {label: '内部页面', value: 3},
- ],
- },
- // 具体值
- value: {
- val: '',
- init: '',
- msg: '',
- state: 0,
- showText: '',// 展示用字段
- oldShowText: '',
- },
- // file
- fileData: {
- val: '',
- init: '',
- msg: '',
- state: 0,
- showText: '',// 展示用字段
- oldShowText: '',
- },
- },
- productSelectVisible: false,
- productSearch: {
- type: {
- val: pTypes[0].key,
- oldVal: pTypes[0].key,
- init: pTypes[0].key,
- msg: '',
- options: pTypes,
- }
- },
- newsSelectVisible: false,
- newsVisible: false,
- newsSearch: {
- type: {
- val: newsType[0].key,
- oldVal: newsType[0].key,
- init: newsType[0].key,
- msg: '',
- options: pTypes,
- }
- },
- imageSelectVisible: false,
- }
- },
- mounted() {
- if(this.carouselList.length === 0){
- this.getCarouselList();
- }
- },
- methods: {
- async getCarouselList(){
- this.loading = true;
- let [err,res] = await handle(this.$axios.get(apiMap.carouselList.path));
- this.loading = false;
- if(err){
- if(this.NotificationKey){
- this.$notification.close(this.NotificationKey);
- }
- this.NotificationKey = `open${Date.now()}`;
- return this.$notification.error({
- message: '轮播数据加载失败',
- description:`异常: ${err.message}`,
- duration: 0,
- btn: h => {
- return h(
- 'a-button',
- {
- props: {
- type: 'primary',
- size: 'small',
- },
- on: {
- click: () => {
- this.$notification.close(this.NotificationKey);
- this.getCarouselList();
- },
- },
- },
- '重试',
- );
- },
- key:this.NotificationKey,
- onClose: close,
- });
- }
- let result = res.data;
- if(result.code === rCode.OK){
- this.carouselList = result.data;
- return {carouselList: result.data}
- }else{
- this.$message.error(result.msg);
- return {}
- }
- },
- // 搜索产品,只需要产品名等信息
- async getProductSearch(searchParam){
- console.log(searchParam)
- if(this.productSearch.type.val !== this.productSearch.type.oldVal){
- searchParam.p = 1;
- }
- searchParam.type = this.productSearch.type.val;
- searchParam.l = this.limit;
- searchParam.p = searchParam.page;
- let [err,res] = await handle(
- this.$axios.get(
- apiMap.searchProductMini.path,
- {params:searchParam})
- );
- if(err){
- console.log(err);
- return [{message:'请求数据失败'},null];
- }
- let result = res.data;
- if(result.code === rCode.OK){
- this.productSearch.type.oldVal = this.productSearch.type.val;
- // data 转换
- result.data = result.data.map(item=>{
- item.showText=item.name;
- return item;
- });
- return [null,result];
- }else{
- // 可捕获的服务器错误
- return [{message:result.msg},null];
- }
- },
- // 加载轮播默认数据
- checkFormItem(field,enumOptions,reCheckField){
- let formItem = this.form[field];
- if (formItem){
- if (enumOptions){
- // 遍历枚举
- for (let i = 0; i < enumOptions.length; i++) {
- let enumOption = enumOptions[i];
- if (enumOption.value === formItem.val){
- return true;
- }
- }
- formItem.msg = '选项不在范围内';
- return false;
- }
- if(reCheckField){
- // 检查用字段
- formItem.msg = fieldIsAllow({
- [reCheckField]:formItem.val,
- })
- }else{
- formItem.msg = fieldIsAllow({
- [field]:formItem.val,
- })
- }
- }else{
- let r = true;
- for (const fieldKey in this.form) {
- this.form[fieldKey].msg = fieldIsAllow({
- [fieldKey]:this.form[fieldKey].val,
- })
- if (this.form[fieldKey].msg){
- r = false;
- }
- }
- return r
- }
- },
- initCarouseForm(){
- this.carouselData = {};
- let keys = Object.keys(this.form);
- for(let i = 0; i < keys.length; i++){
- let key = keys[i];
- this.form[key].val = this.form[key].init;
- this.form[key].msg = '';
- this.form[key].state = 0;
- this.form.value.showText = '';
- }
- },
- openAddCarouselModal(){
- // 初始化表单
- this.initCarouseForm();
- // 打开弹窗. 选择图片,填写链接地址,排序
- this.carouselPopShow = true;
- this.isEditCarousel = false;
- },
- showPop(){
- this.popShow = true;
- this.popLoading = false;
- },
- cancelPop(){
- this.imageSelectVisible = false;
- },
- okHandle(fileItem){
- console.log('文件列表');
- console.log(fileItem);
- this.cancelPop();
- this.$nextTick(()=>{
- this.form.fileData.val = fileItem.filePath;
- this.form.fileData.state = 1;
- this.form.fileData.msg = '';
- this.form.fileData.showText = fileItem.filePath;
- })
- },
- onProductSearchHandle(e){
- console.log(`onProductSearchHandle ${e}`);
- console.log(e);
- console.log(this.productSearch.type.val);
- // this.productSearch.type
- },
- onSelectedItemHandle(item){
- console.log(`selected item ${item}`);
- console.log(item);
- this.form.value.val = item.id;
- this.form.value.showText = item.showText;
- },
- onTypeChangeHandle(e){
- console.log(`type change ${e}`);
- // 清除其他值
- this.form.value.msg = '';
- this.form.value.showText = '';
- },
- loadDefaultProductItem(){
- this.productSearch.type.val = pTypes[0].key;
- this.$refs.productSearch.loadData(1,true);
- },
- },
- })
- </script>
- <template>
- <div class="w-full p-2">
- <rounded-title>轮播图管理</rounded-title>
- <div class="mt-2 rounded bg-white p-2">
- <!-- 轮播图list , 左侧轮播图片, 右侧 轮播信息 -->
- <div class="mt-2 rounded bg-white p-2">
- <div class="py-1 border-b border-cyan-300 flex justify-between">
- 点击下方快进行管理轮播图数据,一次性不要添加过多轮播图
- <!-- 新增按钮-->
- <a-button type="primary" class="ant-icon-btn" icon="plus" @click="openAddCarouselModal" :loading="loading"></a-button>
- <!-- 刷新按钮-->
- <a-button type="primary" class="ant-icon-btn" icon="reload" @click="getCarouselList" :loading="loading"></a-button>
- </div>
- <div class="w-full h-auto transition">
- <div v-show="loading" class="w-full h-64 flex justify-center items-center ">
- <a-spin size="large" />
- </div>
- <div
- v-for="(item,index) in carouselList"
- :key="'carouse-'+index"
- class="mt-2 rounded border flex h-72 overflow-hidden"
- >
- <div class="media w-1/2 h-full">
- <image-viewer :src="item.filePath"></image-viewer>
- </div>
- <div class="w-1/2 h-full box-border pl-2">
- <a-button @click="showPop">编辑</a-button>
- 排序{{item.sort}},数字越小越靠前
- 链接地址: {{item.href}}
- </div>
- </div>
- </div>
- </div>
- </div>
- <pop :show="carouselPopShow" :loading="carouselPopLoading">
- <pop-card>
- <template slot="header" class="w-full">
- {{isEditCarousel ? '编辑轮播图' : '新增轮播图'}}
- </template>
- <template slot="close-group">
- <a-button icon="close" @click="carouselPopShow = false"></a-button>
- </template>
- <div class="w-full">
- <input-row :msg="form.sort.msg"
- label="排序">
- <a-input-number v-model="form.sort.val"
- @focus="form.password.msg=''"
- @blur="checkFormItem('sort')"
- />
- </input-row>
- <!-- 轮播类型选择 -->
- <input-row :msg="form.type.msg"
- label="轮播类型">
- <a-radio-group v-model="form.type.val" @change="onTypeChangeHandle">
- <a-radio-button v-for="opt in form.type.options"
- :key="'cType'+opt.value"
- :value="opt.value">
- {{ opt.label }}
- </a-radio-button>
- </a-radio-group>
- </input-row>
- <!-- 轮播具体值 -->
- <!-- 链接-->
- <input-row
- v-show="form.type.val === dbField_esm.db_base.carouselType.href"
- :msg="form.value.msg"
- label="输入链接">
- <a-input v-model="form.value.val"
- placeholder="输入要指向的链接地址"
- @focus="form.value.msg=''"
- @blur="checkFormItem('value', null,'href')"
- />
- </input-row>
- <!-- 产品选择-->
- <input-row
- v-show="form.type.val === dbField_esm.db_base.carouselType.production"
- :msg="form.value.msg"
- label="选择产品">
- {{ form.value.showText }}
- <a-popover v-model="productSelectVisible" title="选择产品" trigger="click">
- <div slot="content" class="searchBox" >
- <search-box
- class="h-72"
- :loadDataApi="getProductSearch"
- :limit="limit"
- search-placeholder="请输入产品名称关键字"
- loadTip="正在搜索产品中..."
- ref="productSearch"
- @onSelectedItem="onSelectedItemHandle"
- >
- <div class="w-full" slot="otherSearchItem">
- <a-radio-group v-model="productSearch.type.val"
- @change="onProductSearchHandle">
- <a-radio-button v-for="opt in productSearch.type.options"
- :key="'cType'+opt.key"
- :value="opt.key">
- {{ opt.text }}
- </a-radio-button>
- </a-radio-group>
- </div>
- </search-box>
- </div>
- <a-button type="primary" @click="loadDefaultProductItem">
- 选择产品
- </a-button>
- </a-popover>
- </input-row>
- <!-- 新闻选择-->
- <input-row
- v-show="form.type.val === dbField_esm.db_base.carouselType.production"
- :msg="form.value.msg"
- label="文章选择">
- {{ form.value.showText }}
- <a-popover v-model="productSelectVisible" title="选择你需要的文章" trigger="click">
- <div slot="content" class="searchBox" >
- <search-box
- class="h-72"
- :loadDataApi="getProductSearch"
- :limit="limit"
- search-placeholder="请输入产品名称关键字"
- loadTip="正在搜索产品中..."
- ref="productSearch"
- @onSelectedItem="onSelectedItemHandle"
- >
- <div class="w-full" slot="otherSearchItem">
- <a-radio-group v-model="productSearch.type.val"
- @change="onProductSearchHandle">
- <a-radio-button v-for="opt in productSearch.type.options"
- :key="'cType'+opt.key"
- :value="opt.key">
- {{ opt.text }}
- </a-radio-button>
- </a-radio-group>
- </div>
- </search-box>
- </div>
- <a-button type="primary" @click="loadDefaultProductItem">
- 选择产品
- </a-button>
- </a-popover>
- </input-row>
- <!-- 选择图片 -->
- <input-row label="轮播图片">
- <a-popover v-model="imageSelectVisible"
- class="w-full"
- trigger="click">
- <image-table slot="content"
- class="w-full h-full"
- @cancel="imageSelectVisible = false"
- @ok="okHandle"></image-table>
- <div class="w-full h-60 rounded relative">
- <image-viewer class="" :src="form.fileData.showText"></image-viewer>
- <div class="absolute w-full h-full left-0 top-0
- justify-center text-white bg-gray-400
- items-center text-2xl flex opacity-0 hover:opacity-70">
- 点击选择图片
- </div>
- </div>
- </a-popover>
- </input-row>
- </div>
- <template class="w-full" slot="footer">
- <a-button>{{isEditCarousel? '保存': '新增'}}</a-button>
- </template>
- </pop-card>
- </pop>
- </div>
- </template>
- <style scoped>
- .searchBox{
- width: 420px;
- height: 520px;
- }
- </style>
|