!266 【antd/ele】【mp】迁移

Merge pull request !266 from hw/reform-mp
This commit is contained in:
芋道源码
2025-11-20 09:57:17 +00:00
committed by Gitee
78 changed files with 353 additions and 569 deletions

View File

@@ -24,6 +24,49 @@ export namespace MpDraftApi {
articles: Article[]; articles: Article[];
createTime?: Date; createTime?: Date;
} }
/** 图文项(包含预览字段) */
export interface NewsItem {
title: string;
thumbMediaId: string;
author: string;
digest: string;
showCoverPic: number;
content: string;
contentSourceUrl: string;
needOpenComment: number;
onlyFansCanComment: number;
thumbUrl: string;
picUrl?: string; // 用于预览封面
}
/** 图文列表 */
export interface NewsItemList {
newsItem: NewsItem[];
}
/** 草稿文章(用于展示) */
export interface DraftArticle {
mediaId: string;
content: NewsItemList;
updateTime: number;
}
}
/** 创建空的图文项 */
export function createEmptyNewsItem(): MpDraftApi.NewsItem {
return {
title: '',
thumbMediaId: '',
author: '',
digest: '',
showCoverPic: 0,
content: '',
contentSourceUrl: '',
needOpenComment: 0,
onlyFansCanComment: 0,
thumbUrl: '',
};
} }
/** 查询草稿列表 */ /** 查询草稿列表 */

View File

@@ -1,15 +1,8 @@
import type { PageParam, PageResult } from '@vben/request'; import type { PageParam, PageResult } from '@vben/request';
import { requestClient } from '#/api/request'; import { MaterialType } from '@vben/constants';
/** 素材类型枚举 */ import { requestClient } from '#/api/request';
// TODO @hwmp 相关的枚举,可以考虑放这里来。
export enum MaterialType {
IMAGE = 1, // 图片
THUMB = 4, // 缩略图
VIDEO = 3, // 视频
VOICE = 2, // 语音
}
export namespace MpMaterialApi { export namespace MpMaterialApi {
/** 素材信息 */ /** 素材信息 */

View File

@@ -1,20 +1,6 @@
import { requestClient } from '#/api/request'; import { MenuType } from '@vben/constants';
/** 菜单类型枚举 */ import { requestClient } from '#/api/request';
// TODO @hwmp 相关的枚举,可以考虑放这里来。
export enum MenuType {
CLICK = 'click', // 点击推事件
LOCATION_SELECT = 'location_select', // 发送位置
MEDIA_ID = 'media_id', // 下发消息
MINIPROGRAM = 'miniprogram', // 小程序
PIC_PHOTO_OR_ALBUM = 'pic_photo_or_album', // 拍照或者相册发图
PIC_SYSPHOTO = 'pic_sysphoto', // 系统拍照发图
PIC_WEIXIN = 'pic_weixin', // 微信相册发图
SCANCODE_PUSH = 'scancode_push', // 扫码推事件
SCANCODE_WAITMSG = 'scancode_waitmsg', // 扫码带提示
VIEW = 'view', // 跳转 URL
VIEW_LIMITED = 'view_limited', // 跳转图文消息URL
}
export namespace MpMenuApi { export namespace MpMenuApi {
/** 菜单按钮信息 */ /** 菜单按钮信息 */

View File

@@ -1,19 +1,8 @@
import type { PageParam, PageResult } from '@vben/request'; import type { PageParam, PageResult } from '@vben/request';
import { requestClient } from '#/api/request'; import { MessageType } from '@vben/constants';
/** 消息类型枚举 */ import { requestClient } from '#/api/request';
// TODO @hwmp 相关的枚举,可以考虑放这里来。
export enum MessageType {
IMAGE = 'image', // 图片消息
MPNEWS = 'mpnews', // 公众号图文消息
MUSIC = 'music', // 音乐消息
NEWS = 'news', // 图文消息
TEXT = 'text', // 文本消息
VIDEO = 'video', // 视频消息
VOICE = 'voice', // 语音消息
WXCARD = 'wxcard', // 卡券消息
}
export namespace MpMessageApi { export namespace MpMessageApi {
/** 消息信息 */ /** 消息信息 */

View File

@@ -3,13 +3,11 @@ import type { VxeGridPropTypes } from '#/adapter/vxe-table';
import { markRaw } from 'vue'; import { markRaw } from 'vue';
import { DICT_TYPE } from '@vben/constants'; import { DICT_TYPE, AutoReplyMsgType as MsgType } from '@vben/constants';
import { getDictOptions } from '@vben/hooks'; import { getDictOptions } from '@vben/hooks';
import { WxReply } from '#/views/mp/components'; import { WxReply } from '#/views/mp/components';
import { MsgType } from './types';
// TODO @芋艿:要不要使用统一枚举? // TODO @芋艿:要不要使用统一枚举?
const RequestMessageTypes = new Set([ const RequestMessageTypes = new Set([
'image', 'image',
@@ -143,10 +141,6 @@ export function useFormSchema(msgType: MsgType): VbenFormSchema[] {
fieldName: 'reply', fieldName: 'reply',
label: '回复消息', label: '回复消息',
component: markRaw(WxReply), component: markRaw(WxReply),
// TODO @hw这里注释要不要删除掉
// componentProps: {
// modelValue: { type: 'video', content: '12456' },
// },
modelPropName: 'modelValue', modelPropName: 'modelValue',
}); });
return schema; return schema;

View File

@@ -5,6 +5,7 @@ import type { MpAutoReplyApi } from '#/api/mp/autoReply';
import { computed, nextTick, ref } from 'vue'; import { computed, nextTick, ref } from 'vue';
import { confirm, DocAlert, Page, useVbenModal } from '@vben/common-ui'; import { confirm, DocAlert, Page, useVbenModal } from '@vben/common-ui';
import { AutoReplyMsgType as MsgType } from '@vben/constants';
import { IconifyIcon } from '@vben/icons'; import { IconifyIcon } from '@vben/icons';
import { message, Row, Tabs } from 'ant-design-vue'; import { message, Row, Tabs } from 'ant-design-vue';
@@ -21,7 +22,6 @@ import { WxAccountSelect } from '#/views/mp/components';
import { useGridColumns, useGridFormSchema } from './data'; import { useGridColumns, useGridFormSchema } from './data';
import ReplyContentCell from './modules/content.vue'; import ReplyContentCell from './modules/content.vue';
import Form from './modules/form.vue'; import Form from './modules/form.vue';
import { MsgType } from './types';
defineOptions({ name: 'MpAutoReply' }); defineOptions({ name: 'MpAutoReply' });

View File

@@ -4,16 +4,15 @@ import type { Reply } from '#/views/mp/components';
import { computed, nextTick, ref } from 'vue'; import { computed, nextTick, ref } from 'vue';
import { useVbenModal } from '@vben/common-ui'; import { useVbenModal } from '@vben/common-ui';
import { AutoReplyMsgType as MsgType, ReplyType } from '@vben/constants';
import { message } from 'ant-design-vue'; import { message } from 'ant-design-vue';
import { useVbenForm } from '#/adapter/form'; import { useVbenForm } from '#/adapter/form';
import { createAutoReply, updateAutoReply } from '#/api/mp/autoReply'; import { createAutoReply, updateAutoReply } from '#/api/mp/autoReply';
import { $t } from '#/locales'; import { $t } from '#/locales';
import { ReplyType } from '#/views/mp/components/constants';
import { useFormSchema } from '../data'; import { useFormSchema } from '../data';
import { MsgType } from '../types';
const emit = defineEmits(['success']); const emit = defineEmits(['success']);
@@ -113,13 +112,6 @@ const [Modal, modalApi] = useVbenModal({
// 编辑:加载数据 // 编辑:加载数据
const rowData = data.row; const rowData = data.row;
const formValues: any = { ...rowData }; const formValues: any = { ...rowData };
// TODO @hw下面要删除掉么注释。
// delete formValues.responseMessageType;
// delete formValues.responseContent;
// delete formValues.responseMediaId;
// delete formValues.responseMediaUrl;
// delete formValues.responseDescription;
// delete formValues.responseArticles;
formValues.reply = { formValues.reply = {
type: rowData.responseMessageType, type: rowData.responseMessageType,
accountId: data.accountId || -1, accountId: data.accountId || -1,

View File

@@ -1,8 +0,0 @@
// 消息类型Follow: 关注时回复Message: 消息回复Keyword: 关键词回复)
// 作为 tab.nameenum 的数字不能随意修改,与 api 参数相关
// TODO @hw可以搞到 biz-mp-enum.ts 里。
export enum MsgType {
Follow = 1,
Keyword = 3,
Message = 2,
}

View File

@@ -1,33 +0,0 @@
// TODO @hw看看要不要迁移到 packages/constants/src/biz-mp-enum.ts
export enum ReplyType {
Image = 'image',
Music = 'music',
News = 'news',
Text = 'text',
Video = 'video',
Voice = 'voice',
}
export enum NewsType {
Draft = '2',
Published = '1',
}
export enum MaterialType {
Image = 'image',
News = 'news',
Video = 'video',
Voice = 'voice',
}
export enum MsgType {
Event = 'event',
Image = 'image',
Link = 'link',
Location = 'location',
Music = 'music',
News = 'news',
Text = 'text',
Video = 'video',
Voice = 'voice',
}

View File

@@ -1,13 +1,9 @@
export * from './constants'; export { default as WxAccountSelect } from './wx-account-select/account-select.vue';
export { default as WxLocation } from './wx-location/wx-location.vue';
export * from './wx-account-select'; export { default as WxMaterialSelect } from './wx-material-select/wx-material-select.vue';
export * from './wx-location'; export { default as WxMsg } from './wx-msg/msg.vue';
export * from './wx-material-select'; export { default as WxMusic } from './wx-music/wx-music.vue';
export * from './wx-msg'; export { default as WxNews } from './wx-news/wx-news.vue';
export * from './wx-music'; export { default as WxReply } from './wx-reply/wx-reply.vue';
export * from './wx-news'; export { default as WxVideoPlayer } from './wx-video-play/wx-video-play.vue';
export * from './wx-reply'; export { default as WxVoicePlayer } from './wx-voice-play/wx-voice-play.vue';
export * from './wx-video-play';
export * from './wx-voice-play';
// TODO @hw要不使用 export { default as WxAccountSelect } from './account-select.vue'; 形式;

View File

@@ -4,6 +4,7 @@ import type { VxeTableGridOptions } from '#/adapter/vxe-table';
import { reactive, ref, watch } from 'vue'; import { reactive, ref, watch } from 'vue';
import { Page } from '@vben/common-ui'; import { Page } from '@vben/common-ui';
import { NewsType } from '@vben/constants';
import { IconifyIcon } from '@vben/icons'; import { IconifyIcon } from '@vben/icons';
import { Button, Pagination, Row, Spin } from 'ant-design-vue'; import { Button, Pagination, Row, Spin } from 'ant-design-vue';
@@ -14,8 +15,6 @@ import { getFreePublishPage } from '#/api/mp/freePublish';
import { getMaterialPage } from '#/api/mp/material'; import { getMaterialPage } from '#/api/mp/material';
import { WxNews, WxVideoPlayer, WxVoicePlayer } from '#/views/mp/components'; import { WxNews, WxVideoPlayer, WxVoicePlayer } from '#/views/mp/components';
import { NewsType } from '../constants';
defineOptions({ name: 'WxMaterialSelect' }); defineOptions({ name: 'WxMaterialSelect' });
const props = withDefaults( const props = withDefaults(

View File

@@ -1,4 +1,5 @@
<script lang="ts" setup> <script lang="ts" setup>
import { MpMsgType as MsgType } from '@vben/constants';
import { IconifyIcon } from '@vben/icons'; import { IconifyIcon } from '@vben/icons';
import { import {
@@ -9,7 +10,6 @@ import {
WxVoicePlayer, WxVoicePlayer,
} from '#/views/mp/components'; } from '#/views/mp/components';
import { MsgType } from '../constants';
import MsgEvent from './msg-event.vue'; import MsgEvent from './msg-event.vue';
defineOptions({ name: 'Msg' }); defineOptions({ name: 'Msg' });

View File

@@ -28,7 +28,7 @@ defineExpose({
<Image <Image
:src="article.picUrl" :src="article.picUrl"
:preview="false" :preview="false"
class="material-img" class="material-img flex items-center justify-center"
/> />
<div class="news-content-title"> <div class="news-content-title">
<span>{{ article.title }}</span> <span>{{ article.title }}</span>

View File

@@ -3,14 +3,13 @@ import type { Reply } from './types';
import { computed, ref } from 'vue'; import { computed, ref } from 'vue';
import { NewsType } from '@vben/constants';
import { IconifyIcon } from '@vben/icons'; import { IconifyIcon } from '@vben/icons';
import { Button, Col, Modal, Row } from 'ant-design-vue'; import { Button, Col, Modal, Row } from 'ant-design-vue';
import { WxMaterialSelect, WxNews } from '#/views/mp/components'; import { WxMaterialSelect, WxNews } from '#/views/mp/components';
import { NewsType } from '../constants';
defineOptions({ name: 'TabNews' }); defineOptions({ name: 'TabNews' });
const props = defineProps<{ const props = defineProps<{

View File

@@ -1,6 +1,6 @@
import type { Ref } from 'vue'; import type { Ref } from 'vue';
import type { ReplyType } from '../constants'; import type { ReplyType } from '@vben/constants';
import { unref } from 'vue'; import { unref } from 'vue';

View File

@@ -12,11 +12,11 @@ import type { Reply } from './types';
import { computed, ref, unref, watch } from 'vue'; import { computed, ref, unref, watch } from 'vue';
import { NewsType, ReplyType } from '@vben/constants';
import { IconifyIcon } from '@vben/icons'; import { IconifyIcon } from '@vben/icons';
import { Row, Tabs } from 'ant-design-vue'; import { Row, Tabs } from 'ant-design-vue';
import { NewsType, ReplyType } from '../constants';
import TabImage from './tab-image.vue'; import TabImage from './tab-image.vue';
import TabMusic from './tab-music.vue'; import TabMusic from './tab-music.vue';
import TabNews from './tab-news.vue'; import TabNews from './tab-news.vue';

View File

@@ -1,7 +1,6 @@
<script lang="ts" setup> <script lang="ts" setup>
import type { Article } from './modules/types';
import type { VxeTableGridOptions } from '#/adapter/vxe-table'; import type { VxeTableGridOptions } from '#/adapter/vxe-table';
import type { MpDraftApi } from '#/api/mp/draft';
import { confirm, DocAlert, Page, useVbenModal } from '@vben/common-ui'; import { confirm, DocAlert, Page, useVbenModal } from '@vben/common-ui';
import { $t } from '@vben/locales'; import { $t } from '@vben/locales';
@@ -9,10 +8,9 @@ import { $t } from '@vben/locales';
import { message } from 'ant-design-vue'; import { message } from 'ant-design-vue';
import { ACTION_ICON, TableAction, useVbenVxeGrid } from '#/adapter/vxe-table'; import { ACTION_ICON, TableAction, useVbenVxeGrid } from '#/adapter/vxe-table';
import { deleteDraft, getDraftPage } from '#/api/mp/draft'; import { createEmptyNewsItem, deleteDraft, getDraftPage } from '#/api/mp/draft';
import { submitFreePublish } from '#/api/mp/freePublish'; import { submitFreePublish } from '#/api/mp/freePublish';
import { WxAccountSelect } from '#/views/mp/components'; import { WxAccountSelect } from '#/views/mp/components';
import { createEmptyNewsItem } from '#/views/mp/draft/modules/types';
import { useGridColumns, useGridFormSchema } from './data'; import { useGridColumns, useGridFormSchema } from './data';
import DraftTableCell from './modules/draft-table.vue'; import DraftTableCell from './modules/draft-table.vue';
@@ -49,7 +47,7 @@ async function handleCreate() {
} }
/** 修改草稿 */ /** 修改草稿 */
async function handleEdit(row: Article) { async function handleEdit(row: MpDraftApi.DraftArticle) {
const formValues = await gridApi.formApi.getValues(); const formValues = await gridApi.formApi.getValues();
const accountId = formValues.accountId; const accountId = formValues.accountId;
if (!accountId) { if (!accountId) {
@@ -67,7 +65,7 @@ async function handleEdit(row: Article) {
} }
/** 删除草稿 */ /** 删除草稿 */
async function handleDelete(row: Article) { async function handleDelete(row: MpDraftApi.DraftArticle) {
const formValues = await gridApi.formApi.getValues(); const formValues = await gridApi.formApi.getValues();
const accountId = formValues.accountId; const accountId = formValues.accountId;
if (!accountId) { if (!accountId) {
@@ -89,7 +87,7 @@ async function handleDelete(row: Article) {
} }
/** 发布草稿 */ /** 发布草稿 */
async function handlePublish(row: Article) { async function handlePublish(row: MpDraftApi.DraftArticle) {
const formValues = await gridApi.formApi.getValues(); const formValues = await gridApi.formApi.getValues();
const accountId = formValues.accountId; const accountId = formValues.accountId;
if (!accountId) { if (!accountId) {
@@ -145,7 +143,7 @@ const [Grid, gridApi] = useVbenVxeGrid({
} }
}); });
return { return {
list: drafts.list as unknown as Article[], list: drafts.list as unknown as MpDraftApi.DraftArticle[],
total: drafts.total, total: drafts.total,
}; };
}, },
@@ -160,8 +158,7 @@ const [Grid, gridApi] = useVbenVxeGrid({
refresh: true, refresh: true,
search: true, search: true,
}, },
// TODO @hw这里有点纠结一般是 MpDraftApi.Article但是一改貌似就 linter 告警了。 } as VxeTableGridOptions<MpDraftApi.DraftArticle>,
} as VxeTableGridOptions<Article>,
}); });
</script> </script>

View File

@@ -1,7 +1,7 @@
<script lang="ts" setup> <script lang="ts" setup>
import type { UploadFile } from 'ant-design-vue'; import type { UploadFile } from 'ant-design-vue';
import type { NewsItem } from './types'; import type { MpDraftApi } from '#/api/mp/draft';
import { computed, inject, reactive, ref } from 'vue'; import { computed, inject, reactive, ref } from 'vue';
@@ -14,16 +14,16 @@ import { UploadType, useBeforeUpload } from '#/utils/useUpload';
const props = defineProps<{ const props = defineProps<{
isFirst: boolean; isFirst: boolean;
modelValue: NewsItem; modelValue: MpDraftApi.NewsItem;
}>(); }>();
const emit = defineEmits<{ const emit = defineEmits<{
(e: 'update:modelValue', v: NewsItem): void; (e: 'update:modelValue', v: MpDraftApi.NewsItem): void;
}>(); }>();
const UPLOAD_URL = `${import.meta.env.VITE_BASE_URL}/admin-api/mp/material/upload-permanent`; // 上传永久素材的地址 const UPLOAD_URL = `${import.meta.env.VITE_BASE_URL}/admin-api/mp/material/upload-permanent`; // 上传永久素材的地址
const HEADERS = { Authorization: `Bearer ${useAccessStore().accessToken}` }; const HEADERS = { Authorization: `Bearer ${useAccessStore().accessToken}` };
const newsItem = computed<NewsItem>({ const newsItem = computed<MpDraftApi.NewsItem>({
get() { get() {
return props.modelValue; return props.modelValue;
}, },

View File

@@ -1,12 +1,12 @@
<script lang="ts" setup> <script lang="ts" setup>
import type { Article } from './types'; import type { MpDraftApi } from '#/api/mp/draft';
import { WxNews } from '#/views/mp/components'; import { WxNews } from '#/views/mp/components';
defineOptions({ name: 'DraftTableCell' }); defineOptions({ name: 'DraftTableCell' });
const props = defineProps<{ const props = defineProps<{
row: Article; row: MpDraftApi.DraftArticle;
}>(); }>();
</script> </script>

View File

@@ -1,11 +1,11 @@
<script lang="ts" setup> <script lang="ts" setup>
import type { NewsItem } from './types'; import type { MpDraftApi } from '#/api/mp/draft';
import { computed, provide, ref } from 'vue'; import { computed, provide, ref } from 'vue';
import { useVbenModal } from '@vben/common-ui'; import { useVbenModal } from '@vben/common-ui';
import { message, Spin } from 'ant-design-vue'; import { message } from 'ant-design-vue';
import { createDraft, updateDraft } from '#/api/mp/draft'; import { createDraft, updateDraft } from '#/api/mp/draft';
@@ -16,10 +16,9 @@ const emit = defineEmits(['success']);
const formData = ref<{ const formData = ref<{
accountId: number; accountId: number;
mediaId?: string; mediaId?: string;
newsList?: NewsItem[]; newsList?: MpDraftApi.NewsItem[];
}>(); }>();
const newsList = ref<NewsItem[]>([]); const newsList = ref<MpDraftApi.NewsItem[]>([]);
const isSubmitting = ref(false);
const getTitle = computed(() => { const getTitle = computed(() => {
return formData.value?.mediaId ? '修改图文' : '新建图文'; return formData.value?.mediaId ? '修改图文' : '新建图文';
@@ -35,9 +34,6 @@ const [Modal, modalApi] = useVbenModal({
if (!formData.value) { if (!formData.value) {
return; return;
} }
// TODO @hw是不是 isSubmitting 非必须哈?因为 modal 已经去 lock 啦。
isSubmitting.value = true;
modalApi.lock(); modalApi.lock();
try { try {
if (formData.value.mediaId) { if (formData.value.mediaId) {
@@ -54,7 +50,6 @@ const [Modal, modalApi] = useVbenModal({
await modalApi.close(); await modalApi.close();
emit('success'); emit('success');
} finally { } finally {
isSubmitting.value = false;
modalApi.unlock(); modalApi.unlock();
} }
}, },
@@ -68,7 +63,7 @@ const [Modal, modalApi] = useVbenModal({
accountId: number; accountId: number;
isCreating: boolean; isCreating: boolean;
mediaId?: string; mediaId?: string;
newsList?: NewsItem[]; newsList?: MpDraftApi.NewsItem[];
}>(); }>();
if (!data) { if (!data) {
return; return;
@@ -85,12 +80,10 @@ const [Modal, modalApi] = useVbenModal({
<template> <template>
<Modal :title="getTitle" class="w-4/5" destroy-on-close> <Modal :title="getTitle" class="w-4/5" destroy-on-close>
<Spin :spinning="isSubmitting">
<NewsForm <NewsForm
v-if="formData" v-if="formData"
v-model="newsList" v-model="newsList"
:is-creating="!formData.mediaId" :is-creating="!formData.mediaId"
/> />
</Spin>
</Modal> </Modal>
</template> </template>

View File

@@ -1,5 +1,5 @@
<script lang="ts" setup> <script lang="ts" setup>
import type { NewsItem } from './types'; import type { MpDraftApi } from '#/api/mp/draft';
import { computed, ref } from 'vue'; import { computed, ref } from 'vue';
@@ -8,23 +8,23 @@ import { IconifyIcon } from '@vben/icons';
import { Button, Col, Input, Layout, Row, Textarea } from 'ant-design-vue'; import { Button, Col, Input, Layout, Row, Textarea } from 'ant-design-vue';
import { createEmptyNewsItem } from '#/api/mp/draft';
import { Tinymce as RichTextarea } from '#/components/tinymce'; import { Tinymce as RichTextarea } from '#/components/tinymce';
import CoverSelect from './cover-select.vue'; import CoverSelect from './cover-select.vue';
import { createEmptyNewsItem } from './types';
defineOptions({ name: 'NewsForm' }); defineOptions({ name: 'NewsForm' });
const props = defineProps<{ const props = defineProps<{
isCreating: boolean; isCreating: boolean;
modelValue: NewsItem[] | null; modelValue: MpDraftApi.NewsItem[] | null;
}>(); }>();
const emit = defineEmits<{ const emit = defineEmits<{
(e: 'update:modelValue', v: NewsItem[]): void; (e: 'update:modelValue', v: MpDraftApi.NewsItem[]): void;
}>(); }>();
const newsList = computed<NewsItem[]>({ const newsList = computed<MpDraftApi.NewsItem[]>({
get() { get() {
return props.modelValue === null return props.modelValue === null
? [createEmptyNewsItem()] ? [createEmptyNewsItem()]

View File

@@ -1,42 +0,0 @@
// TODO @hw这个要融合到 draftApi 里么?类似其他模块的。
interface NewsItem {
title: string;
thumbMediaId: string;
author: string;
digest: string;
showCoverPic: number;
content: string;
contentSourceUrl: string;
needOpenComment: number;
onlyFansCanComment: number;
thumbUrl: string;
picUrl?: string; // 用于预览封面
}
interface NewsItemList {
newsItem: NewsItem[];
}
interface Article {
mediaId: string;
content: NewsItemList;
updateTime: number;
}
function createEmptyNewsItem(): NewsItem {
return {
title: '',
thumbMediaId: '',
author: '',
digest: '',
showCoverPic: 0,
content: '',
contentSourceUrl: '',
needOpenComment: 0,
onlyFansCanComment: 0,
thumbUrl: '',
};
}
export type { Article, NewsItem, NewsItemList };
export { createEmptyNewsItem };

View File

@@ -3,13 +3,13 @@ import type { VxeTableGridOptions } from '#/adapter/vxe-table';
import { onMounted, watch } from 'vue'; import { onMounted, watch } from 'vue';
import { MpMsgType as MsgType } from '@vben/constants';
import { formatDate2 } from '@vben/utils'; import { formatDate2 } from '@vben/utils';
import { Button, Image, Tag } from 'ant-design-vue'; import { Button, Image, Tag } from 'ant-design-vue';
import { useVbenVxeGrid } from '#/adapter/vxe-table'; import { useVbenVxeGrid } from '#/adapter/vxe-table';
import { import {
MsgType,
WxLocation, WxLocation,
WxMusic, WxMusic,
WxNews, WxNews,

View File

@@ -4,7 +4,7 @@ import type { Dayjs } from 'dayjs';
import { reactive, ref } from 'vue'; import { reactive, ref } from 'vue';
import { Page } from '@vben/common-ui'; import { Page } from '@vben/common-ui';
import { DICT_TYPE } from '@vben/constants'; import { DICT_TYPE, MpMsgType as MsgType } from '@vben/constants';
import { getDictOptions } from '@vben/hooks'; import { getDictOptions } from '@vben/hooks';
import { IconifyIcon } from '@vben/icons'; import { IconifyIcon } from '@vben/icons';
@@ -20,7 +20,7 @@ import {
} from 'ant-design-vue'; } from 'ant-design-vue';
import { getMessagePage } from '#/api/mp/message'; import { getMessagePage } from '#/api/mp/message';
import { MsgType, WxAccountSelect, WxMsg } from '#/views/mp/components'; import { WxAccountSelect, WxMsg } from '#/views/mp/components';
import MessageTable from './MessageTable.vue'; import MessageTable from './MessageTable.vue';

View File

@@ -50,13 +50,9 @@ export function updateDraft(
mediaId: string, mediaId: string,
articles: MpDraftApi.Article[], articles: MpDraftApi.Article[],
) { ) {
return requestClient.put( return requestClient.put('/mp/draft/update', articles, {
'/mp/draft/update',
{ articles },
{
params: { accountId, mediaId }, params: { accountId, mediaId },
}, });
);
} }
/** 删除草稿 */ /** 删除草稿 */

View File

@@ -8,7 +8,7 @@ import { DICT_TYPE } from '@vben/constants';
import { getDictObj, getDictOptions } from '@vben/hooks'; import { getDictObj, getDictOptions } from '@vben/hooks';
import { getSimpleAccountList } from '#/api/mp/account'; import { getSimpleAccountList } from '#/api/mp/account';
import { ReplySelect } from '#/views/mp/components'; import { WxReplySelect } from '#/views/mp/components';
import { MsgType } from './modules/types'; import { MsgType } from './modules/types';
@@ -143,7 +143,7 @@ export function useFormSchema(msgType: MsgType): VbenFormSchema[] {
schema.push({ schema.push({
fieldName: 'reply', fieldName: 'reply',
label: '回复消息', label: '回复消息',
component: markRaw(ReplySelect), component: markRaw(WxReplySelect),
}); });
return schema; return schema;
} }

View File

@@ -18,6 +18,7 @@ import {
import { ACTION_ICON, TableAction, useVbenVxeGrid } from '#/adapter/vxe-table'; import { ACTION_ICON, TableAction, useVbenVxeGrid } from '#/adapter/vxe-table';
import * as MpAutoReplyApi from '#/api/mp/autoReply'; import * as MpAutoReplyApi from '#/api/mp/autoReply';
import { $t } from '#/locales'; import { $t } from '#/locales';
import { WxAccountSelect } from '#/views/mp/components';
import { useGridColumns, useGridFormSchema } from './data'; import { useGridColumns, useGridFormSchema } from './data';
import Content from './modules/content.vue'; import Content from './modules/content.vue';
@@ -27,6 +28,12 @@ import { MsgType } from './modules/types';
defineOptions({ name: 'MpAutoReply' }); defineOptions({ name: 'MpAutoReply' });
const msgType = ref<string>(String(MsgType.Keyword)); // 消息类型 const msgType = ref<string>(String(MsgType.Keyword)); // 消息类型
/** 公众号变化时查询数据 */
function handleAccountChange(accountId: number) {
gridApi.formApi.setValues({ accountId });
gridApi.formApi.submitForm();
}
/** 切换回复类型 */ /** 切换回复类型 */
async function onTabChange(tabName: string) { async function onTabChange(tabName: string) {
msgType.value = tabName; msgType.value = tabName;
@@ -91,8 +98,6 @@ const [FormModal, formModalApi] = useVbenModal({
const [Grid, gridApi] = useVbenVxeGrid({ const [Grid, gridApi] = useVbenVxeGrid({
formOptions: { formOptions: {
schema: useGridFormSchema(), schema: useGridFormSchema(),
// 表单值变化时自动提交,这样 accountId 会被正确传递到查询函数
submitOnChange: true,
}, },
gridOptions: { gridOptions: {
columns: useGridColumns(Number(msgType.value) as MsgType), columns: useGridColumns(Number(msgType.value) as MsgType),
@@ -109,6 +114,7 @@ const [Grid, gridApi] = useVbenVxeGrid({
}); });
}, },
}, },
autoLoad: false,
}, },
rowConfig: { rowConfig: {
keyField: 'id', keyField: 'id',
@@ -148,6 +154,9 @@ const showCreateButton = computed(() => {
<FormModal @success="handleRefresh" /> <FormModal @success="handleRefresh" />
<Grid table-title="自动回复列表"> <Grid table-title="自动回复列表">
<template #form-accountId>
<WxAccountSelect @change="handleAccountChange" />
</template>
<!-- 在工具栏上方放置 Tab 切换 --> <!-- 在工具栏上方放置 Tab 切换 -->
<template #toolbar-actions> <template #toolbar-actions>
<ElTabs <ElTabs

View File

@@ -1,5 +1,10 @@
<script lang="ts" setup> <script lang="ts" setup>
import { Music, News, VideoPlayer, VoicePlayer } from '#/views/mp/components'; import {
WxMusic,
WxNews,
WxVideoPlayer,
WxVoicePlayer,
} from '#/views/mp/components';
defineOptions({ name: 'ReplyContentCell' }); defineOptions({ name: 'ReplyContentCell' });
@@ -14,7 +19,7 @@ const props = defineProps<{
{{ props.row.responseContent }} {{ props.row.responseContent }}
</div> </div>
<div v-else-if="props.row.responseMessageType === 'voice'"> <div v-else-if="props.row.responseMessageType === 'voice'">
<VoicePlayer <WxVoicePlayer
v-if="props.row.responseMediaUrl" v-if="props.row.responseMediaUrl"
:url="props.row.responseMediaUrl" :url="props.row.responseMediaUrl"
/> />
@@ -30,17 +35,17 @@ const props = defineProps<{
props.row.responseMessageType === 'shortvideo' props.row.responseMessageType === 'shortvideo'
" "
> >
<VideoPlayer <WxVideoPlayer
v-if="props.row.responseMediaUrl" v-if="props.row.responseMediaUrl"
:url="props.row.responseMediaUrl" :url="props.row.responseMediaUrl"
class="mt-[10px]" class="mt-[10px]"
/> />
</div> </div>
<div v-else-if="props.row.responseMessageType === 'news'"> <div v-else-if="props.row.responseMessageType === 'news'">
<News :articles="props.row.responseArticles" /> <WxNews :articles="props.row.responseArticles" />
</div> </div>
<div v-else-if="props.row.responseMessageType === 'music'"> <div v-else-if="props.row.responseMessageType === 'music'">
<Music <WxMusic
:title="props.row.responseTitle" :title="props.row.responseTitle"
:description="props.row.responseDescription" :description="props.row.responseDescription"
:thumb-media-url="props.row.responseThumbMediaUrl" :thumb-media-url="props.row.responseThumbMediaUrl"

View File

@@ -1,5 +1,5 @@
<script lang="ts" setup> <script lang="ts" setup>
import type { Reply } from '#/views/mp/components/reply/types'; import type { Reply } from '#/views/mp/components/wx-reply/types';
import { computed, nextTick, ref } from 'vue'; import { computed, nextTick, ref } from 'vue';
@@ -10,7 +10,7 @@ import { ElMessage } from 'element-plus';
import { useVbenForm } from '#/adapter/form'; import { useVbenForm } from '#/adapter/form';
import { createAutoReply, updateAutoReply } from '#/api/mp/autoReply'; import { createAutoReply, updateAutoReply } from '#/api/mp/autoReply';
import { $t } from '#/locales'; import { $t } from '#/locales';
import { ReplyType } from '#/views/mp/components/reply/types'; import { ReplyType } from '#/views/mp/components/wx-reply/types';
import { useFormSchema } from '../data'; import { useFormSchema } from '../data';
import { MsgType } from './types'; import { MsgType } from './types';

View File

@@ -1 +0,0 @@
export { default } from './account-select.vue';

View File

@@ -1,24 +1,22 @@
// 统一导出所有模块组件 // 统一导出所有模块组件
export { default as AccountSelect } from './account-select/account-select.vue'; export { default as WxAccountSelect } from './wx-account-select/wx-account-select.vue';
export { default as WxAccountSelect } from './account-select/account-select.vue';
// TODO @hw还是带着 wx 前缀。。。貌似好点,我的锅!!! export { default as WxLocation } from './wx-location/wx-location.vue';
export { default as Location } from './location/location.vue'; export * from './wx-material-select/types';
export { default as MaterialSelect } from './material-select/material-select.vue';
export * from './material-select/types'; export { default as WxMaterialSelect } from './wx-material-select/wx-material-select.vue';
export * from './msg/types'; export * from './wx-msg/types';
export { default as Music } from './music/music.vue'; export { default as WxMusic } from './wx-music/wx-music.vue';
export { default as News } from './news/news.vue'; export { default as WxNews } from './wx-news/wx-news.vue';
export { default as ReplySelect } from './reply/reply.vue'; export * from './wx-reply/types';
export * from './reply/types'; export { default as WxReplySelect } from './wx-reply/wx-reply.vue';
export { default as VideoPlayer } from './video-play/video-play.vue'; export { default as WxVideoPlayer } from './wx-video-play/wx-video-play.vue';
export { default as VoicePlayer } from './voice-play/voice-play.vue'; export { default as WxVoicePlayer } from './wx-voice-play/wx-voice-play.vue';

View File

@@ -1 +0,0 @@
export { default } from './location.vue';

View File

@@ -1,3 +0,0 @@
export { default } from './material-select.vue';
export { MaterialType, NewsType } from './types';

View File

@@ -1,3 +0,0 @@
export { default } from './msg.vue';
export { MsgType } from './types';

View File

@@ -1 +0,0 @@
export { default } from './music.vue';

View File

@@ -1 +0,0 @@
export { default } from './news.vue';

View File

@@ -1 +0,0 @@
export { default } from './video-play.vue';

View File

@@ -1 +0,0 @@
export { default } from './voice-play.vue';

View File

@@ -0,0 +1 @@
export { default } from './wx-account-select.vue';

View File

@@ -0,0 +1 @@
export { default } from './wx-location.vue';

View File

@@ -0,0 +1,3 @@
export { MaterialType, NewsType } from './types';
export { default } from './wx-material-select.vue';

View File

@@ -20,9 +20,9 @@ import {
import * as MpDraftApi from '#/api/mp/draft'; import * as MpDraftApi from '#/api/mp/draft';
import * as MpFreePublishApi from '#/api/mp/freePublish'; import * as MpFreePublishApi from '#/api/mp/freePublish';
import * as MpMaterialApi from '#/api/mp/material'; import * as MpMaterialApi from '#/api/mp/material';
import News from '#/views/mp/components/news/news.vue'; import News from '#/views/mp/components/wx-news/wx-news.vue';
import VideoPlayer from '#/views/mp/components/video-play/video-play.vue'; import VideoPlayer from '#/views/mp/components/wx-video-play/wx-video-play.vue';
import VoicePlayer from '#/views/mp/components/voice-play/voice-play.vue'; import VoicePlayer from '#/views/mp/components/wx-voice-play/wx-voice-play.vue';
import { NewsType } from './types'; import { NewsType } from './types';

View File

@@ -0,0 +1,3 @@
export { MsgType } from './types';
export { default } from './wx-msg.vue';

View File

@@ -5,7 +5,7 @@ import { formatDateTime } from '@vben/utils';
import avatarWechat from '#/assets/imgs/wechat.png'; import avatarWechat from '#/assets/imgs/wechat.png';
import Msg from './msg.vue'; import Msg from './wx-msg.vue';
// User 使 // User 使
type PropsUser = User; type PropsUser = User;

View File

@@ -1,11 +1,11 @@
<script lang="ts" setup> <script lang="ts" setup>
import { ref } from 'vue'; import { ref } from 'vue';
import Location from '#/views/mp/components/location/location.vue'; import Location from '#/views/mp/components/wx-location/wx-location.vue';
import Music from '#/views/mp/components/music/music.vue'; import Music from '#/views/mp/components/wx-music/wx-music.vue';
import News from '#/views/mp/components/news/news.vue'; import News from '#/views/mp/components/wx-news/wx-news.vue';
import VideoPlayer from '#/views/mp/components/video-play/video-play.vue'; import VideoPlayer from '#/views/mp/components/wx-video-play/wx-video-play.vue';
import VoicePlayer from '#/views/mp/components/voice-play/voice-play.vue'; import VoicePlayer from '#/views/mp/components/wx-voice-play/wx-voice-play.vue';
import { MsgType } from '../types'; import { MsgType } from '../types';
import MsgEvent from './msg-event.vue'; import MsgEvent from './msg-event.vue';

View File

@@ -0,0 +1 @@
export { default } from './wx-music.vue';

View File

@@ -63,5 +63,5 @@ defineExpose({
<style lang="scss" scoped> <style lang="scss" scoped>
/* 因为 joolun 实现依赖 avue 组件,该页面使用了 card.scss */ /* 因为 joolun 实现依赖 avue 组件,该页面使用了 card.scss */
@import url('../msg/card.scss'); @import url('../wx-msg/card.scss');
</style> </style>

View File

@@ -0,0 +1 @@
export { default } from './wx-news.vue';

View File

@@ -6,6 +6,8 @@
代码优化补充注释提升阅读性 代码优化补充注释提升阅读性
--> -->
<script lang="ts" setup> <script lang="ts" setup>
import { ElImage } from 'element-plus';
defineOptions({ name: 'WxNews' }); defineOptions({ name: 'WxNews' });
const props = withDefaults( const props = withDefaults(
@@ -28,11 +30,10 @@ defineExpose({
<!-- 头条 --> <!-- 头条 -->
<a v-if="index === 0" :href="article.url" target="_blank"> <a v-if="index === 0" :href="article.url" target="_blank">
<div class="news-main"> <div class="news-main">
<div class="news-content"> <div class="news-content flex items-center justify-center">
<el-image <ElImage
:src="article.picUrl || article.thumbUrl" :src="article.picUrl || article.thumbUrl"
class="material-img" class="material-img"
style="width: 100%; height: 120px"
/> />
<div class="news-content-title"> <div class="news-content-title">
<span>{{ article.title }}</span> <span>{{ article.title }}</span>
@@ -45,7 +46,7 @@ defineExpose({
<div class="news-main-item"> <div class="news-main-item">
<div class="news-content-item"> <div class="news-content-item">
<div class="news-content-item-title">{{ article.title }}</div> <div class="news-content-item-title">{{ article.title }}</div>
<div class="news-content-item-img"> <div class="news-content-item-img flex items-center justify-center">
<img <img
:src="article.picUrl || article.thumbUrl" :src="article.picUrl || article.thumbUrl"
class="material-img" class="material-img"
@@ -118,6 +119,9 @@ defineExpose({
} }
.material-img { .material-img {
display: flex;
align-items: center;
justify-content: center;
width: 100%; width: 100%;
} }
</style> </style>

View File

@@ -1,3 +1,3 @@
export { default } from './reply.vue';
export { createEmptyReply, type Reply, ReplyType } from './types'; export { createEmptyReply, type Reply, ReplyType } from './types';
export { default } from './wx-reply.vue';

View File

@@ -18,7 +18,7 @@ import {
} from 'element-plus'; } from 'element-plus';
import { UploadType, useBeforeUpload } from '#/utils/useUpload'; import { UploadType, useBeforeUpload } from '#/utils/useUpload';
import MaterialSelect from '#/views/mp/components/material-select/material-select.vue'; import MaterialSelect from '#/views/mp/components/wx-material-select/wx-material-select.vue';
const props = defineProps<{ const props = defineProps<{
modelValue: Reply; modelValue: Reply;

View File

@@ -20,7 +20,7 @@ import {
import { UploadType, useBeforeUpload } from '#/utils/useUpload'; import { UploadType, useBeforeUpload } from '#/utils/useUpload';
// import { getAccessToken } from '@/utils/auth' // import { getAccessToken } from '@/utils/auth'
import MaterialSelect from '#/views/mp/components/material-select/material-select.vue'; import MaterialSelect from '#/views/mp/components/wx-material-select/wx-material-select.vue';
// //

View File

@@ -7,10 +7,10 @@ import { IconifyIcon } from '@vben/icons';
import { ElButton, ElCol, ElDialog, ElRow } from 'element-plus'; import { ElButton, ElCol, ElDialog, ElRow } from 'element-plus';
import MaterialSelect from '#/views/mp/components/material-select/material-select.vue'; import MaterialSelect from '#/views/mp/components/wx-material-select/wx-material-select.vue';
import News from '#/views/mp/components/news/news.vue'; import News from '#/views/mp/components/wx-news/wx-news.vue';
import { NewsType } from '../material-select/types'; import { NewsType } from '../wx-material-select/types';
const props = defineProps<{ const props = defineProps<{
modelValue: Reply; modelValue: Reply;

View File

@@ -19,8 +19,8 @@ import {
} from 'element-plus'; } from 'element-plus';
import { UploadType, useBeforeUpload } from '#/utils/useUpload'; import { UploadType, useBeforeUpload } from '#/utils/useUpload';
import MaterialSelect from '#/views/mp/components/material-select/material-select.vue'; import MaterialSelect from '#/views/mp/components/wx-material-select/wx-material-select.vue';
import VideoPlayer from '#/views/mp/components/video-play/video-play.vue'; import VideoPlayer from '#/views/mp/components/wx-video-play/wx-video-play.vue';
const props = defineProps<{ const props = defineProps<{
modelValue: Reply; modelValue: Reply;

View File

@@ -18,8 +18,8 @@ import {
} from 'element-plus'; } from 'element-plus';
import { UploadType, useBeforeUpload } from '#/utils/useUpload'; import { UploadType, useBeforeUpload } from '#/utils/useUpload';
import MaterialSelect from '#/views/mp/components/material-select/material-select.vue'; import MaterialSelect from '#/views/mp/components/wx-material-select/wx-material-select.vue';
import VoicePlayer from '#/views/mp/components/voice-play/voice-play.vue'; import VoicePlayer from '#/views/mp/components/wx-voice-play/wx-voice-play.vue';
// //

View File

@@ -16,7 +16,7 @@ import { IconifyIcon } from '@vben/icons';
import { ElRow, ElTabPane, ElTabs } from 'element-plus'; import { ElRow, ElTabPane, ElTabs } from 'element-plus';
import { NewsType } from '../material-select/types'; import { NewsType } from '../wx-material-select/types';
import TabImage from './tab-image.vue'; import TabImage from './tab-image.vue';
import TabMusic from './tab-music.vue'; import TabMusic from './tab-music.vue';
import TabNews from './tab-news.vue'; import TabNews from './tab-news.vue';

View File

@@ -0,0 +1 @@
export { default } from './wx-video-play.vue';

View File

@@ -0,0 +1 @@
export { default } from './wx-voice-play.vue';

View File

@@ -1,10 +1,5 @@
import type { VbenFormSchema } from '#/adapter/form'; import type { VbenFormSchema } from '#/adapter/form';
import type { VxeTableGridOptions } from '#/adapter/vxe-table'; import type { VxeTableGridOptions } from '#/adapter/vxe-table';
import { markRaw } from 'vue';
import AccountSelect from '#/views/mp/components/account-select/account-select.vue';
/** 获取表格列配置 */ /** 获取表格列配置 */
export function useGridColumns(): VxeTableGridOptions['columns'] { export function useGridColumns(): VxeTableGridOptions['columns'] {
return [ return [
@@ -14,12 +9,6 @@ export function useGridColumns(): VxeTableGridOptions['columns'] {
minWidth: 300, minWidth: 300,
slots: { default: 'content' }, slots: { default: 'content' },
}, },
{
field: 'updateTime',
title: '更新时间',
minWidth: 180,
formatter: 'formatDateTime',
},
{ {
title: '操作', title: '操作',
width: 200, width: 200,
@@ -35,7 +24,7 @@ export function useGridFormSchema(): VbenFormSchema[] {
{ {
fieldName: 'accountId', fieldName: 'accountId',
label: '公众号', label: '公众号',
component: markRaw(AccountSelect), component: 'Input',
}, },
]; ];
} }

View File

@@ -3,15 +3,14 @@ import type { Article } from './modules/types';
import type { VxeTableGridOptions } from '#/adapter/vxe-table'; import type { VxeTableGridOptions } from '#/adapter/vxe-table';
import { nextTick, onMounted, provide, ref, watch } from 'vue';
import { DocAlert, Page, useVbenModal } from '@vben/common-ui'; import { DocAlert, Page, useVbenModal } from '@vben/common-ui';
import { ElLoading, ElMessage, ElMessageBox } from 'element-plus'; import { ElLoading, ElMessage, ElMessageBox } from 'element-plus';
import { ACTION_ICON, TableAction, useVbenVxeGrid } from '#/adapter/vxe-table'; import { ACTION_ICON, TableAction, useVbenVxeGrid } from '#/adapter/vxe-table';
import * as MpDraftApi from '#/api/mp/draft'; import { deleteDraft, getDraftPage } from '#/api/mp/draft';
import * as MpFreePublishApi from '#/api/mp/freePublish'; import * as MpFreePublishApi from '#/api/mp/freePublish';
import { WxAccountSelect } from '#/views/mp/components';
import { createEmptyNewsItem } from '#/views/mp/draft/modules/types'; import { createEmptyNewsItem } from '#/views/mp/draft/modules/types';
import { useGridColumns, useGridFormSchema } from './data'; import { useGridColumns, useGridFormSchema } from './data';
@@ -25,10 +24,20 @@ const [FormModal, formModalApi] = useVbenModal({
destroyOnClose: true, destroyOnClose: true,
}); });
/** 刷新表格 */
function handleRefresh() {
gridApi.query();
}
/** 公众号变化时查询数据 */
function handleAccountChange(accountId: number) {
gridApi.formApi.setValues({ accountId });
gridApi.formApi.submitForm();
}
const [Grid, gridApi] = useVbenVxeGrid({ const [Grid, gridApi] = useVbenVxeGrid({
formOptions: { formOptions: {
schema: useGridFormSchema(), schema: useGridFormSchema(),
submitOnChange: true,
}, },
gridOptions: { gridOptions: {
columns: useGridColumns(), columns: useGridColumns(),
@@ -37,68 +46,23 @@ const [Grid, gridApi] = useVbenVxeGrid({
proxyConfig: { proxyConfig: {
ajax: { ajax: {
query: async ({ page }, formValues) => { query: async ({ page }, formValues) => {
// 更新 accountId const drafts = await getDraftPage({
if (formValues?.accountId) {
accountId.value = formValues.accountId;
}
const drafts = await MpDraftApi.getDraftPage({
pageNo: page.currentPage, pageNo: page.currentPage,
pageSize: page.pageSize, pageSize: page.pageSize,
...formValues, ...formValues,
}); });
// 处理 API 返回的数据,兼容不同的数据结构 // 将 thumbUrl 转成 picUrl保证 wx-news 组件可以预览封面
const formattedList: Article[] = drafts.list.map((draft: any) => { drafts.list.forEach((draft: any) => {
// 如果已经是 content.newsItem 格式,直接使用 const newsList = draft.content?.newsItem;
if (draft.content?.newsItem) { if (newsList) {
const newsItem = draft.content.newsItem.map((item: any) => ({ newsList.forEach((item: any) => {
...item, item.picUrl = item.thumbUrl || item.picUrl;
picUrl: item.thumbUrl || item.picUrl, });
}));
return {
mediaId: draft.mediaId,
content: {
newsItem,
},
updateTime:
draft.updateTime ||
(draft.createTime
? new Date(draft.createTime).getTime()
: Date.now()),
};
} }
// 如果是 articles 格式,转换为 content.newsItem 格式
if (draft.articles) {
const newsItem = draft.articles.map((article: any) => ({
...article,
thumbUrl: article.thumbUrl || article.thumbMediaId,
picUrl: article.thumbUrl || article.thumbMediaId,
}));
return {
mediaId: draft.mediaId,
content: {
newsItem,
},
updateTime:
draft.updateTime ||
(draft.createTime
? new Date(draft.createTime).getTime()
: Date.now()),
};
}
// 默认返回空结构
return {
mediaId: draft.mediaId || '',
content: {
newsItem: [],
},
updateTime: draft.updateTime || Date.now(),
};
}); });
return { return {
page: { list: drafts.list as unknown as Article[],
total: drafts.total, total: drafts.total,
},
result: formattedList,
}; };
}, },
}, },
@@ -115,21 +79,6 @@ const [Grid, gridApi] = useVbenVxeGrid({
} as VxeTableGridOptions<Article>, } as VxeTableGridOptions<Article>,
}); });
// 提供 accountId 给子组件
const accountId = ref<number>(-1);
// 监听表单提交,更新 accountId
watch(
() => gridApi.formApi?.getLatestSubmissionValues?.()?.accountId,
(newAccountId) => {
if (newAccountId !== undefined) {
accountId.value = newAccountId;
}
},
);
provide('accountId', accountId);
/** 新增按钮操作 */ /** 新增按钮操作 */
async function handleCreate() { async function handleCreate() {
const formValues = await gridApi.formApi.getValues(); const formValues = await gridApi.formApi.getValues();
@@ -162,7 +111,7 @@ async function handleEdit(row: Article) {
isCreating: false, isCreating: false,
accountId, accountId,
mediaId: row.mediaId, mediaId: row.mediaId,
newsList: structuredClone(row.content.newsItem), newsList: row.content.newsItem,
}) })
.open(); .open();
} }
@@ -201,7 +150,7 @@ async function handlePublish(row: Article) {
async function handleDelete(row: Article) { async function handleDelete(row: Article) {
const formValues = await gridApi.formApi.getValues(); const formValues = await gridApi.formApi.getValues();
const accountId = formValues.accountId; const accountId = formValues.accountId;
if (!accountId || accountId === -1) { if (!accountId) {
ElMessage.warning('请先选择公众号'); ElMessage.warning('请先选择公众号');
return; return;
} }
@@ -212,9 +161,9 @@ async function handleDelete(row: Article) {
text: '删除中...', text: '删除中...',
}); });
try { try {
await MpDraftApi.deleteDraft(accountId, row.mediaId); await deleteDraft(accountId, row.mediaId);
ElMessage.success('删除成功'); ElMessage.success('删除成功');
await gridApi.query(); handleRefresh();
} finally { } finally {
loadingInstance.close(); loadingInstance.close();
} }
@@ -222,19 +171,6 @@ async function handleDelete(row: Article) {
// //
} }
} }
// 页面挂载后,等待表单初始化完成再加载数据
onMounted(async () => {
await nextTick();
if (gridApi.formApi) {
const formValues = await gridApi.formApi.getValues();
if (formValues.accountId) {
accountId.value = formValues.accountId;
gridApi.formApi.setLatestSubmissionValues(formValues);
await gridApi.query();
}
}
});
</script> </script>
<template> <template>
@@ -243,15 +179,12 @@ onMounted(async () => {
<DocAlert title="公众号图文" url="https://doc.iocoder.cn/mp/article/" /> <DocAlert title="公众号图文" url="https://doc.iocoder.cn/mp/article/" />
</template> </template>
<FormModal <FormModal @success="handleRefresh" />
@success="
() => {
gridApi.query();
}
"
/>
<Grid table-title="草稿列表"> <Grid table-title="草稿列表">
<template #form-accountId>
<WxAccountSelect @change="handleAccountChange" />
</template>
<template #toolbar-tools> <template #toolbar-tools>
<TableAction <TableAction
:actions="[ :actions="[
@@ -310,7 +243,12 @@ onMounted(async () => {
.vxe-table--body { .vxe-table--body {
.vxe-body--column { .vxe-body--column {
.vxe-cell { .vxe-cell {
height: auto !important;
padding: 0; padding: 0;
img {
width: 300px !important;
}
} }
} }
} }

View File

@@ -11,7 +11,7 @@ import { useAccessStore } from '@vben/stores';
import { ElButton, ElDialog, ElImage, ElMessage, ElUpload } from 'element-plus'; import { ElButton, ElDialog, ElImage, ElMessage, ElUpload } from 'element-plus';
import { UploadType, useBeforeUpload } from '#/utils/useUpload'; import { UploadType, useBeforeUpload } from '#/utils/useUpload';
import MaterialSelect from '#/views/mp/components/material-select/material-select.vue'; import MaterialSelect from '#/views/mp/components/wx-material-select/wx-material-select.vue';
// 设置上传的请求头部 // 设置上传的请求头部
@@ -80,20 +80,22 @@ function onUploadError(err: Error) {
<template> <template>
<div> <div>
<p>封面:</p> <p>封面:</p>
<div class="thumb-div"> <div
class="inline-flex w-full flex-col items-center justify-center text-center"
>
<ElImage <ElImage
v-if="newsItem.thumbUrl" v-if="newsItem.thumbUrl"
style="width: 300px; max-height: 300px" class="max-h-[300px] w-[300px]"
:src="newsItem.thumbUrl" :src="newsItem.thumbUrl"
fit="contain" fit="contain"
/> />
<IconifyIcon <IconifyIcon
v-else v-else
icon="ep:plus" icon="ep:plus"
class="avatar-uploader-icon" class="border border-[#d9d9d9] text-center text-[28px] leading-[120px] text-[#8c939d]"
:class="isFirst ? 'avatar' : 'avatar1'" :class="isFirst ? 'h-[120px] w-[230px]' : 'h-[120px] w-[120px]'"
/> />
<div class="thumb-but"> <div class="m-1.5">
<ElUpload <ElUpload
:action="UPLOAD_URL" :action="UPLOAD_URL"
:headers="HEADERS" :headers="HEADERS"
@@ -112,12 +114,12 @@ function onUploadError(err: Error) {
size="small" size="small"
type="primary" type="primary"
@click="showImageDialog = true" @click="showImageDialog = true"
style="margin-left: 5px" class="ml-1.5"
> >
素材库选择 素材库选择
</ElButton> </ElButton>
<template #tip> <template #tip>
<div class="el-upload__tip"> <div class="ml-1.5">
支持 bmp/png/jpeg/jpg/gif 格式大小不超过 2M 支持 bmp/png/jpeg/jpg/gif 格式大小不超过 2M
</div> </div>
</template> </template>
@@ -139,43 +141,3 @@ function onUploadError(err: Error) {
</div> </div>
</div> </div>
</template> </template>
<style lang="scss" scoped>
.el-upload__tip {
margin-left: 5px;
}
.thumb-div {
display: inline-block;
display: flex;
flex-direction: column;
align-items: center;
justify-content: center;
width: 100%;
text-align: center;
.avatar-uploader-icon {
width: 120px;
height: 120px;
font-size: 28px;
line-height: 120px;
color: #8c939d;
text-align: center;
border: 1px solid #d9d9d9;
}
.avatar {
width: 230px;
height: 120px;
}
.avatar1 {
width: 120px;
height: 120px;
}
.thumb-but {
margin: 5px;
}
}
</style>

View File

@@ -1,7 +1,7 @@
<script lang="ts" setup> <script lang="ts" setup>
import type { Article } from './types'; import type { Article } from './types';
import News from '#/views/mp/components/news/news.vue'; import News from '#/views/mp/components/wx-news/wx-news.vue';
defineOptions({ name: 'DraftTableCell' }); defineOptions({ name: 'DraftTableCell' });
@@ -11,15 +11,9 @@ const props = defineProps<{
</script> </script>
<template> <template>
<div class="draft-content"> <div class="p-2.5">
<div v-if="props.row.content && props.row.content.newsItem"> <div v-if="props.row.content && props.row.content.newsItem">
<News :articles="props.row.content.newsItem" /> <News :articles="props.row.content.newsItem" />
</div> </div>
</div> </div>
</template> </template>
<style lang="scss" scoped>
.draft-content {
padding: 10px;
}
</style>

View File

@@ -1,7 +1,7 @@
<script lang="ts" setup> <script lang="ts" setup>
import type { NewsItem } from './types'; import type { NewsItem } from './types';
import { computed, ref } from 'vue'; import { computed, provide, ref } from 'vue';
import { useVbenModal } from '@vben/common-ui'; import { useVbenModal } from '@vben/common-ui';
@@ -27,6 +27,11 @@ const getTitle = computed(() => {
return formData.value?.isCreating ? '新建图文' : '修改图文'; return formData.value?.isCreating ? '新建图文' : '修改图文';
}); });
// 提供 accountId 给子组件
provide(
'accountId',
computed(() => formData.value?.accountId),
);
const [Modal, modalApi] = useVbenModal({ const [Modal, modalApi] = useVbenModal({
async onConfirm() { async onConfirm() {
if (!formData.value) { if (!formData.value) {

View File

@@ -98,19 +98,28 @@ function plusNews() {
<template> <template>
<ElContainer> <ElContainer>
<ElAside width="40%"> <ElAside width="40%">
<div class="select-item"> <div class="mx-auto mb-2.5 w-3/5 border border-gray-200 p-2.5">
<div v-for="(news, index) in newsList" :key="index"> <div v-for="(news, index) in newsList" :key="index">
<div <div
class="news-main father" class="group mx-auto h-[120px] w-full cursor-pointer bg-white"
v-if="index === 0" v-if="index === 0"
:class="{ activeAddNews: activeNewsIndex === index }" :class="{
'border-[5px] border-[#2bb673]': activeNewsIndex === index,
}"
@click="activeNewsIndex = index" @click="activeNewsIndex = index"
> >
<div class="news-content"> <div class="relative h-[120px] w-full bg-[#acadae]">
<img class="material-img" :src="news.thumbUrl" /> <img class="h-full w-full" :src="news.thumbUrl" />
<div class="news-content-title">{{ news.title }}</div> <div
class="absolute bottom-0 left-0 inline-block h-[25px] w-[98%] overflow-hidden text-ellipsis whitespace-nowrap bg-black p-[1%] text-[15px] text-white opacity-65"
>
{{ news.title }}
</div> </div>
<div class="child" v-if="newsList.length > 1"> </div>
<div
v-if="newsList.length > 1"
class="relative -bottom-6 hidden text-center group-hover:block"
>
<ElButton <ElButton
type="info" type="info"
circle circle
@@ -131,18 +140,22 @@ function plusNews() {
</div> </div>
</div> </div>
<div <div
class="news-main-item father" class="group mx-auto w-full cursor-pointer border-t border-gray-200 bg-white py-1.5"
v-if="index > 0" v-if="index > 0"
:class="{ activeAddNews: activeNewsIndex === index }" :class="{
'border-[5px] border-[#2bb673]': activeNewsIndex === index,
}"
@click="activeNewsIndex = index" @click="activeNewsIndex = index"
> >
<div class="news-content-item"> <div class="relative -ml-0.5">
<div class="news-content-item-title">{{ news.title }}</div> <div class="inline-block w-[70%] text-xs">{{ news.title }}</div>
<div class="news-content-item-img"> <div class="inline-block w-1/4 bg-[#acadae]">
<img class="material-img" :src="news.thumbUrl" width="100%" /> <img class="h-full w-full" :src="news.thumbUrl" width="100%" />
</div> </div>
</div> </div>
<div class="child"> <div
class="relative -bottom-6 hidden text-center group-hover:block"
>
<ElButton <ElButton
v-if="newsList.length > index + 1" v-if="newsList.length > index + 1"
circle circle
@@ -173,7 +186,10 @@ function plusNews() {
</div> </div>
</div> </div>
</div> </div>
<ElRow justify="center" class="ope-row"> <ElRow
justify="center"
class="mt-1.5 border-t border-gray-200 pt-1.5 text-center"
>
<ElButton <ElButton
type="primary" type="primary"
circle circle
@@ -188,7 +204,7 @@ function plusNews() {
<ElMain> <ElMain>
<div v-if="newsList.length > 0 && activeNewsItem"> <div v-if="newsList.length > 0 && activeNewsItem">
<!-- 标题作者原文地址 --> <!-- 标题作者原文地址 -->
<ElRow :gutter="20"> <ElRow :gutter="20" class="mb-5 last:mb-0">
<ElInput <ElInput
v-model="activeNewsItem.title" v-model="activeNewsItem.title"
placeholder="请输入标题(必填)" placeholder="请输入标题(必填)"
@@ -196,16 +212,16 @@ function plusNews() {
<ElInput <ElInput
v-model="activeNewsItem.author" v-model="activeNewsItem.author"
placeholder="请输入作者" placeholder="请输入作者"
style="margin-top: 5px" class="mt-1.5"
/> />
<ElInput <ElInput
v-model="activeNewsItem.contentSourceUrl" v-model="activeNewsItem.contentSourceUrl"
placeholder="请输入原文地址" placeholder="请输入原文地址"
style="margin-top: 5px" class="mt-1.5"
/> />
</ElRow> </ElRow>
<!-- 封面和摘要 --> <!-- 封面和摘要 -->
<ElRow :gutter="20"> <ElRow :gutter="20" class="mb-5 last:mb-0">
<ElCol :span="12"> <ElCol :span="12">
<CoverSelect <CoverSelect
v-model="activeNewsItem" v-model="activeNewsItem"
@@ -219,123 +235,16 @@ function plusNews() {
type="textarea" type="textarea"
v-model="activeNewsItem.digest" v-model="activeNewsItem.digest"
placeholder="请输入摘要" placeholder="请输入摘要"
class="digest" class="inline-block w-full align-top"
maxlength="120" maxlength="120"
/> />
</ElCol> </ElCol>
</ElRow> </ElRow>
<!--富文本编辑器组件--> <!--富文本编辑器组件-->
<ElRow> <ElRow class="mb-5 last:mb-0">
<RichTextarea v-model="activeNewsItem.content" /> <RichTextarea v-model="activeNewsItem.content" />
</ElRow> </ElRow>
</div> </div>
</ElMain> </ElMain>
</ElContainer> </ElContainer>
</template> </template>
<style lang="scss" scoped>
.ope-row {
padding-top: 5px;
margin-top: 5px;
text-align: center;
border-top: 1px solid #eaeaea;
}
.el-row {
margin-bottom: 20px;
}
.el-row:last-child {
margin-bottom: 0;
}
.digest {
display: inline-block;
width: 100%;
vertical-align: top;
}
/* 新增图文 */
.news-main {
width: 100%;
height: 120px;
margin: auto;
background-color: #fff;
}
.news-content {
position: relative;
width: 100%;
height: 120px;
background-color: #acadae;
}
.news-content-title {
position: absolute;
bottom: 0;
left: 0;
display: inline-block;
width: 98%;
height: 25px;
padding: 1%;
overflow: hidden;
text-overflow: ellipsis;
font-size: 15px;
color: #fff;
white-space: nowrap;
background-color: black;
opacity: 0.65;
}
.news-main-item {
width: 100%;
padding: 5px 0;
margin: auto;
background-color: #fff;
border-top: 1px solid #eaeaea;
}
.news-content-item {
position: relative;
margin-left: -3px;
}
.news-content-item-title {
display: inline-block;
width: 70%;
font-size: 12px;
}
.news-content-item-img {
display: inline-block;
width: 25%;
background-color: #acadae;
}
.select-item {
width: 60%;
padding: 10px;
margin: 0 auto 10px;
border: 1px solid #eaeaea;
.activeAddNews {
border: 5px solid #2bb673;
}
}
.father .child {
position: relative;
bottom: 25px;
display: none;
text-align: center;
}
.father:hover .child {
display: block;
}
.material-img {
width: 100%;
height: 100%;
}
</style>

View File

@@ -14,9 +14,9 @@ import {
ElSelect, ElSelect,
} from 'element-plus'; } from 'element-plus';
import MaterialSelect from '#/views/mp/components/material-select/material-select.vue'; import MaterialSelect from '#/views/mp/components/wx-material-select/wx-material-select.vue';
import News from '#/views/mp/components/news/news.vue'; import News from '#/views/mp/components/wx-news/wx-news.vue';
import ReplySelect from '#/views/mp/components/reply/reply.vue'; import ReplySelect from '#/views/mp/components/wx-reply/wx-reply.vue';
import menuOptions from './menuOptions'; import menuOptions from './menuOptions';

View File

@@ -1 +1,78 @@
// TODO @hwmp 相关的枚举,可以考虑放这里来。 /** 回复类型枚举 */
export enum ReplyType {
Image = 'image',
Music = 'music',
News = 'news',
Text = 'text',
Video = 'video',
Voice = 'voice',
}
/** 图文类型枚举 */
export enum NewsType {
Draft = '2',
Published = '1',
}
/** 回复素材类型枚举 */
export enum ReplyMaterialType {
Image = 'image',
News = 'news',
Video = 'video',
Voice = 'voice',
}
/** 微信消息类型枚举 */
export enum MpMsgType {
Event = 'event',
Image = 'image',
Link = 'link',
Location = 'location',
Music = 'music',
News = 'news',
Text = 'text',
Video = 'video',
Voice = 'voice',
}
/** 自动回复消息类型枚举Follow: 关注时回复Message: 消息回复Keyword: 关键词回复) */
export enum AutoReplyMsgType {
Follow = 1,
Keyword = 3,
Message = 2,
}
/** 消息类型枚举 */
export enum MessageType {
IMAGE = 'image', // 图片消息
MPNEWS = 'mpnews', // 公众号图文消息
MUSIC = 'music', // 音乐消息
NEWS = 'news', // 图文消息
TEXT = 'text', // 文本消息
VIDEO = 'video', // 视频消息
VOICE = 'voice', // 语音消息
WXCARD = 'wxcard', // 卡券消息
}
/** 素材类型枚举 */
export enum MaterialType {
IMAGE = 1, // 图片
THUMB = 4, // 缩略图
VIDEO = 3, // 视频
VOICE = 2, // 语音
}
/** 菜单类型枚举 */
export enum MenuType {
CLICK = 'click', // 点击推事件
LOCATION_SELECT = 'location_select', // 发送位置
MEDIA_ID = 'media_id', // 下发消息
MINIPROGRAM = 'miniprogram', // 小程序
PIC_PHOTO_OR_ALBUM = 'pic_photo_or_album', // 拍照或者相册发图
PIC_SYSPHOTO = 'pic_sysphoto', // 系统拍照发图
PIC_WEIXIN = 'pic_weixin', // 微信相册发图
SCANCODE_PUSH = 'scancode_push', // 扫码推事件
SCANCODE_WAITMSG = 'scancode_waitmsg', // 扫码带提示
VIEW = 'view', // 跳转 URL
VIEW_LIMITED = 'view_limited', // 跳转图文消息URL
}