This commit is contained in:
zhangxingye 2025-02-22 16:25:05 +08:00
parent d215d1185e
commit 93d0a89a20
92 changed files with 7180 additions and 268 deletions

View File

@ -90,3 +90,11 @@ export function cancelInstall(addon: string) {
export function getInstalledAddonList() {
return request.get('addon/list/install')
}
/**
*
* @returns
*/
export function getShowApp() {
return request.get('addon/list/showapp')
}

View File

@ -1,5 +1,6 @@
import request from '@/utils/request'
/**
*
* @param params

View File

@ -197,4 +197,33 @@ export function getApps(params: Record<string, any>) {
*/
export function copyDiy(params: Record<string, any>) {
return request.post(`diy/copy`, params, { showSuccessMessage: true })
}
/***************************************************** 主题风格 ****************************************************/
/**
*
* @param params
*/
export function getDefaultTheme(params: Record<string, any>) {
return request.get(`diy/theme/color`, {params})
}
/**
*
* @param params
*/
export function getDiyTheme(params: Record<string, any>) {
return request.get(`diy/theme`, {params})
}
/**
*
* @param params
*/
export function setDiyTheme(params: Record<string, any>) {
return request.post(`diy/theme`, params, { showSuccessMessage: true })
}

View File

@ -0,0 +1,228 @@
import request from '@/utils/request'
/***************************************************** 万能表单 ****************************************************/
/**
*
* @param params
* @returns
*/
export function getDiyFormPageList(params: Record<string, any>) {
return request.get(`diy/form`, { params })
}
/**
*
* @param params
* @returns
*/
export function getDiyFormList(params: Record<string, any>) {
return request.get(`diy/form/list`, { params })
}
/**
*
* @param form_id id
* @returns
*/
export function getDiyFormInfo(form_id: number) {
return request.get(`diy/form/${ form_id }`);
}
/**
*
* @param params
* @returns
*/
export function addDiyForm(params: Record<string, any>) {
return request.post('diy/form', params, { showSuccessMessage: true })
}
/**
*
* @param params
*/
export function editDiyForm(params: Record<string, any>) {
return request.put(`diy/form/${ params.form_id }`, params, { showSuccessMessage: true })
}
/**
*
* @param params
*/
export function editDiyFormShare(params: Record<string, any>) {
return request.put(`diy/form/share`, params, { showSuccessMessage: true })
}
/**
*
* @param params
* @returns
*/
export function deleteDiyForm(params: Record<string, any>) {
return request.put(`diy/form/delete`, params, { showSuccessMessage: true })
}
/**
*
*/
export function initPage(params: Record<string, any>) {
return request.get(`diy/form/init`, { params })
}
/**
*
* @param params
* @returns
*/
export function getDiyFormQrcode(params: Record<string, any>) {
return request.get(`diy/form/qrcode`, { params })
}
/**
*
* @param params
* @returns
*/
export function getDiyFormFieldsList(params: Record<string, any>) {
return request.get(`diy/form/fields/list`, { params })
}
/**
*
* @param params
* @returns
*/
export function getDiyFormFieldStat(params: Record<string, any>) {
return request.get(`diy/form/records/field/stat`, { params })
}
/**
*
*/
export function getDiyTemplate(params: Record<string, any>) {
return request.get(`diy/template`, { params })
}
/**
*
*/
export function getDiyTemplatePages(params: Record<string, any>) {
return request.get(`diy/form/template`, { params })
}
/**
*
* @param params
* @returns
*/
export function editFormStatus(params: Record<string, any>) {
return request.put(`diy/form/status`, params, {
showErrorMessage: true,
showSuccessMessage: true
})
}
/**
*
* @param params
* @returns
*/
export function getApps(params: Record<string, any>) {
return request.get(`diy/apps`)
}
/**
*
* @param params
*/
export function copyDiy(params: Record<string, any>) {
return request.post(`diy/form/copy`, params, { showSuccessMessage: true })
}
/**
*
* @param params
* @returns
*/
export function getFormType(params: Record<string, any>) {
return request.get(`diy/form/type`)
}
/**
*
* @param form_id
* @returns
*/
export function getFormWriteConfig(form_id: any) {
return request.get(`diy/form/write/${ form_id }`)
}
/**
*
* @param params
*/
export function editDiyFormWriteConfig(params: Record<string, any>) {
return request.put(`diy/form/write`, params, { showSuccessMessage: true })
}
/**
*
* @param form_id
* @returns
*/
export function getFormSubmitConfig(form_id: any) {
return request.get(`diy/form/submit/${ form_id }`)
}
/**
*
* @param params
*/
export function editDiyFormSubmitConfig(params: Record<string, any>) {
return request.put(`diy/form/submit`, params, { showSuccessMessage: true })
}
/**
*
* @param params
* @returns
*/
export function getFormRecords(params: Record<string, any>) {
return request.get(`diy/form/records`, { params })
}
/**
*
* @param id
* @returns
*/
export function getFormRecordsInfo(id: number) {
return request.get(`diy/form/records/${ id }`);
}
/**
*
* @param params
* @returns
*/
export function deleteFormRecords(params: Record<string, any>) {
return request.put(`diy/form/records/delete`, params, { showSuccessMessage: true })
}
/**
*
* @param params
* @returns
*/
export function getFormRecordsMember(params: Record<string, any>) {
return request.get(`diy/form/records/member/stat`, { params })
}
/**
*
* @param params
*/
export function copyForm(params: Record<string, any>) {
return request.post(`diy/form/copy`, params, { showSuccessMessage: true })
}

View File

@ -368,6 +368,21 @@ export function memberTransfer(params: Record<string, any>) {
return request.put(`member/cash_out/transfer/${params.id}`, params, { showSuccessMessage: true })
}
/**
*
* @param params
*/
export function memberRemark(params: Record<string, any>) {
return request.put(`member/cash_out/remark/${params.id}`, params, { showSuccessMessage: true })
}
/**
*
* @param params
*/
export function memberCheck(id: number) {
return request.put(`member/cash_out/check/${id}`, {}, { showSuccessMessage: true })
}
/**
*
* @param params

View File

@ -10,6 +10,15 @@ export function getNoticeList(params: any) {
return request.get('notice/notice', {params})
}
/**
*
* @param params
* @returns
*/
export function getNoticeLog(params: any) {
return request.get(`notice/log`, {params})
}
/**
*
* @param key

View File

@ -91,10 +91,11 @@ export function pay(params: Record<string, any>) {
}
/**
*
* @param params
* @returns
*/
*
* @param tradeType
* @param tradeId
* @param channel
*/
export function getFriendsPay(tradeType : string, tradeId : number, channel: string) {
return request.get(`pay/friendspay/info/${tradeType}/${tradeId}/${channel}`, { showErrorMessage: false })
}
}

View File

@ -391,7 +391,7 @@ export function getTransferInfo(channel: string) {
* @returns
*/
export function setTransferInfo(params: Record<string, any>) {
return request.post(`pay/channel/set/transfer`, params)
return request.post(`pay/channel/set/transfer`, params, { showSuccessMessage: true })
}
/***************************************************** 定时任务 ****************************************************/
@ -570,7 +570,6 @@ export function setPatConfig(params: Record<string, any>) {
return request.post(`pay/channel/set/all`, params, { showSuccessMessage: true })
}
/***************************************************** 刷新菜单 ****************************************************/
/**
*
@ -610,8 +609,6 @@ export function getMap() {
return request.get(`sys/config/map`)
}
/***************************************************** 首页 ****************************************************/
/**
*
*/

Binary file not shown.

After

Width:  |  Height:  |  Size: 3.5 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 167 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 8.8 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 24 KiB

After

Width:  |  Height:  |  Size: 30 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 7.7 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 838 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 964 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 880 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 1011 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 8.2 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 8.9 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 957 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 7.9 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 8.9 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 80 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 5.1 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 466 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 602 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.3 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 34 KiB

After

Width:  |  Height:  |  Size: 24 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.3 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 26 KiB

After

Width:  |  Height:  |  Size: 33 KiB

View File

@ -134,6 +134,7 @@ const emits = defineEmits(['complete', 'cloudbuild'])
const upgradeTipsShowDialog = ref<boolean>(false)
let upgradeLog: any = []
let errorLog: any = []
/**
* 查询升级任务
*/
@ -158,12 +159,18 @@ const getUpgradeTaskFn = () => {
})
//
if (data.error) {
upgradeTask.value = data
ElMessage({ message: '升级失败', type: 'error' })
terminalRef.value.pushMessage({ content: data.error, class: 'error' })
data.error.forEach(item => {
if (!errorLog.includes(item)) {
terminalRef.value.pushMessage({ content: item, class: 'error' })
errorLog.push(item)
}
})
}
//
if (data.step == 'restoreComplete') {
return
}
//
//
if (data.step == 'upgradeComplete') {
active.value = 'complete'
notificationEl && notificationEl.close()
@ -316,6 +323,7 @@ const clearUpgradeTaskFn = () => {
uploading.value = false
upgradeTask.value = null
upgradeLog = []
errorLog = []
flashInterval && clearInterval(flashInterval)
clearUpgradeTask().then(() => {}).catch()
}

View File

@ -1,33 +1,33 @@
{
"menuName": "菜单名称",
"menuType": "类型",
"authId": "权限标识",
"authId": "api路径",
"menuTypeDir": "目录",
"menuTypeMenu": "菜单",
"menuTypeButton": "按钮",
"menuDeleteTips": "确定要删除该菜单吗?",
"addMenu": "添加菜单",
"menuDeleteTips": "删除菜单会删除当前菜单以及该菜单下所有子菜单,是否确认删除?",
"initializeMenu":"重置菜单",
"initializeMenuTipsOne":"重置菜单会将应用或插件的dict目录下的菜单配置文件中菜单配置更新到数据库一般用做开发者修改了dict菜单配置文件后同步到数据库操作。",
"initializeMenuTipsTwo":"如果用户手动调整过以下菜单项,通常允许进行本项操作,操作会重置为原始菜单。 请谨慎使用!",
"addMenu": "添加菜单",
"updateMenu": "编辑菜单",
"routePath": "路由路径",
"viewPath": "组件路径",
"addon":"选择应用",
"parentMenu": "父级菜单",
"menuIcon": "菜单图标",
"sort":"权重",
"menuKey":"菜单标识",
"menuKey":"权限标识",
"menuNamePlaceholder": "请输入菜单名称",
"menuKeyPlaceholder": "请输入菜单标识",
"menuKeyPlaceholder": "请输入权限标识",
"menuKeyValidata":"菜单标识只能使用字母数字下划线并且开头不能为数字",
"routePathPlaceholder": "请输入路由路径",
"viewPathPlaceholder": "请输入组件路径",
"authIdPlaceholder": "请输入权限标识",
"authIdPlaceholder": "请输入api路径",
"selectIconPlaceholder": "请选择菜单图标",
"topLevel": "顶级",
"menuShortName":"菜单短标题",
"menuShortNamePlaceholder":"请输入菜单短标题",
"addon":"选择应用",
"system":"系统菜单",
"application":"应用菜单"
}
}

View File

@ -0,0 +1,394 @@
{
"templatePagePlaceholder": "选择模板",
"templatePageEmpty": "无",
"changeTemplatePageTips": "切换模板后,当前页面内容将被替换且不被保存,请谨慎操作",
"developTitle": "开发环境配置",
"wapDomain": "wap域名WAP_DOMAIN",
"wapDomainPlaceholder": "请输入wap域名",
"pageSet": "页面设置",
"tabEditContent": "内容",
"tabEditStyle": "样式",
"pageStyle": "页面样式",
"pageContent": "页面内容",
"statusBarContent": "导航栏内容",
"statusBarStyle": "导航栏样式",
"statusBarSwitchTips": "此处控制当前页面导航栏是否显示",
"bottomNavContent": "底部导航内容",
"diyPageTitle": "页面名称",
"diyPageTitlePlaceholder": "请输入页面名称",
"pageTitleTips": "页面名称用于后台显示",
"diyTitle": "页面标题",
"diyTitlePlaceholder": "请输入页面标题",
"titleTips": "页面标题用于前台显示",
"pageBgColor": "页面颜色",
"bgUrl": "背景图片",
"bgHeightScale": "高度比例",
"bgHeightScaleTip": "为0时背景高度自适应展示",
"marginSet": "边距设置",
"componentStyleTitle": "组件样式",
"bottomBgColor": "底部背景",
"bottomBgTips": "底部背景包含边距和圆角",
"componentBgColor": "组件背景色",
"componentBgUrl": "组件背景图",
"componentBgAlpha": "透明度",
"bgGradientAngle": "渐变角度",
"topToBottom": "从上到下",
"leftToRight": "从左到右",
"marginTop": "上边距",
"marginBottom": "下边距",
"marginBoth": "左右边距",
"topRounded": "上圆角",
"bottomRounded": "下圆角",
"warmPrompt": "温馨提示",
"leavePageTitleTips": "确定离开此页面?",
"leavePageContentTips": "系统可能不会保存您所做的更改。",
"decorating": "正在装修",
"preview": "保存并预览",
"moveUpComponent": "上移",
"moveDownComponent": "下移",
"copyComponent": "复制",
"delComponent": "删除",
"resetComponent": "重置",
"tabbar": "底部导航",
"tabbarSwitchTips": "此处控制当前页面底部导航菜单是否显示",
"link": "链接地址",
"delComponentTips": "确认要删除当前组件吗?",
"notCopy": "无法复制",
"componentCanOnlyAdd": "组件只能添加",
"piece": "个",
"componentNotMoved": "该组件禁止移动",
"resetComponentTips": "确认要重置组件默认数据吗?",
"image": "图片上传",
"imageUpload": "图片上传",
"imageSet": "图片设置",
"imageAdsTips": "建议上传尺寸相同的图片推荐尺寸750*350",
"imageAdsSameScreenTips": "开启沉浸式样式,请确保该图片广告组件在页面中位于最顶端;为保证体验,请不要开导航栏;沉浸式样式仅在微信小程序中生效。",
"sameScreen": "沉浸式",
"addImageAd": "添加图片",
"imageUrlTip": "请上传图片",
"imageHeight": "图片高度",
"imageHeightPlaceholder": "请输入图片高度",
"imageHeightRegNum": "图片高度格式错误,请输入数字",
"dataSources": "数据来源",
"defaultSources": "默认",
"manualSelectionSources": "手动选择",
"selectPlaceholder": "请选择",
"selected": "已选",
"graphicNavModeTitle": "导航模式",
"layoutMode": "排版模式",
"layoutModeHorizontal": "横排",
"layoutModeVertical": "竖排",
"graphicNavSelectMode": "选择模式",
"graphicNavModeGraphic": "图文导航",
"graphicNavModeImg": "图片导航",
"graphicNavModeText": "文字导航",
"graphicNavImageSet": "图片设置",
"graphicNavImageSize": "图片大小",
"graphicNavAroundRadius": "图片圆角",
"graphicNavShowStyle": "展示风格",
"graphicNavStyleFixed": "固定显示",
"graphicNavStyleSingleSlide": "单行滑动",
"graphicNavStyleMultiLine": "多行滑动",
"graphicNavStylePageSlide": "分页滑动",
"graphicNavRowCount": "每行数量",
"graphicNavPageCount": "显示方式",
"graphicNavSetLabel": "导航设置",
"singleLine": "单行",
"multiline": "多行",
"graphicNavTips": "建议上传尺寸相同的图片推荐尺寸60*60",
"graphicNavTitle": "标题",
"graphicNavTitlePlaceholder": "请输入标题",
"subGraphicNavTitle": "副标题",
"subGraphicNavTitlePlaceholder": "请输入副标题",
"subGraphicNavTitleLink": "副标题链接",
"addGraphicNav": "添加导航",
"blankHeightSet": "高度设置",
"blankHeight": "空白高度",
"styleSet": "风格设置",
"titleStyle": "标题样式",
"selectStyle": "风格选择",
"activeCubeBlockBtnText": "按钮文字",
"btnTextItalics": "斜体",
"btnTextNormal": "常规",
"styleLabel": "风格",
"styleShowTips": "风格 1 2 3仅在小程序中展示",
"titleContent": "标题内容",
"title": "标题名称",
"titlePlaceholder": "请输入标题",
"textAlign": "对齐方式",
"textAlignLeft": "居左",
"textAlignCenter": "居中",
"textAlignRight": "居右",
"textSet": "文字设置",
"textFontSize": "文字大小",
"textFontWeight": "文字粗细",
"fontWeightBold": "加粗",
"fontWeightNormal": "常规",
"textColor": "文字颜色",
"subTitleStyle": "副标题样式",
"subTextBgColor": "背景色",
"subTitleContent": "标题内容",
"subTitle": "副标题",
"subTitlePlaceholder": "请输入副标题",
"moreContent": "“更多”按钮内容",
"more": "文字",
"morePlaceholder": "请输入文字",
"moreIsShow": "是否显示",
"memberStyle": "会员样式",
"template": "模板",
"imageGap": "图片间隙",
"rubikCubeStyle": "魔方样式",
"rubikCubeLayout": "魔方布局",
"hotArea": "热区",
"hotAreaSet": "热区设置",
"hotAreaBackground": "热区背景",
"addHotArea": "添加热区",
"clickSet": "点击设置",
"selectedAfterHotArea": "个热区",
"hotAreaManage": "热区管理",
"selectedHotArea": "请选择热区",
"hotAreaLink": "的链接地址",
"addonListSet": "应用设置",
"addonListTips": "应用选择",
"selectAddonTips": "请选择应用",
"addonTitle": "应用名称",
"addonDesc": "应用描述",
"addonIcon": "应用图标",
"selectAddon": "选择应用",
"addAddon": "添加应用",
"show": "显示",
"hidden": "隐藏",
"goodsCategoryTitle": "商品分类",
"customGoods": "手动选择",
"goodsNum": "商品数量",
"selectCategory": "选择分类",
"categoryName": "分类名称",
"categoryImage": "分类图片",
"selectSource": "选择数据源",
"richTextContentSet": "内容设置",
"richTextPlaceholder": "请输入富文本内容",
"activeCubeBlockContent": "板块内容",
"activeCubeTitle": "标题",
"activeCubeTitlePlaceholder": "请输入标题",
"activeCubeSubTitle": "副标题",
"activeCubeSubTitlePlaceholder": "请输入副标题",
"activeCubeButton": "按钮",
"activeCubeButtonPlaceholder": "请输入按钮文字",
"activeCubeButtonColor": "按钮颜色",
"activeListFrameColor": "框体颜色",
"activeCubeSubTitleTextColor": "文字颜色",
"activeCubeSubTitleBgColor": "背景颜色",
"activeCubeAddItem": "添加一个板块",
"activeCubeBlockStyle": "板块样式",
"activeCubeBlockTextFontWeight": "标题粗细",
"noticeStyle": "公告风格",
"noticeType": "类型",
"noticeTypeImg": "图片",
"noticeTypeText": "文字",
"noticeTypeTextPlaceholder": "请输入公告标题",
"noticeTitle": "公告标题",
"addNotice": "添加公告",
"noticeText": "公告内容",
"noticeScrollWay": "滚动方式",
"noticeUpDown": "上下滚动",
"noticeHorizontal": "横向滚动",
"noticeShowType": "点击类型",
"noticeShowPopUp": "弹出公告内容",
"noticeShowLink": "跳转链接",
"dragMouseAdjustOrder": "鼠标拖拽可调整顺序",
"noticePlaceholderText": "请输入公告内容",
"carouselSearchShowPosition": "显示设置",
"carouselSearchOpen": "开启",
"carouselSearchClose": "关闭",
"carouselSearchBgGradient": "背景渐变",
"carouselSearchShowWay": "展示方式",
"carouselSearchShowWayStatic": "正常显示",
"carouselSearchShowWayFixed": "滚动至顶部固定",
"carouselSearchFixedBgColor": "置顶背景",
"carouselSearchStyleSelect": "风格选择",
"carouselSearchSet": "搜索设置",
"carouselSearchSubTitle": "副标题",
"carouselSearchSubTitleStyle": "副标题样式",
"carouselSearchPositionStyle": "定位样式",
"carouselSearchSubTitlePlaceholder": "请输入副标题内容",
"carouselSearchText": "搜索内容",
"carouselSearchTextColor": "文字颜色",
"carouselSearchBgColor": "背景颜色",
"carouselSearchBtnColor": "按钮颜色",
"carouselSearchBtnBgColor": "按钮背景色",
"carouselSearchHotWordSet": "搜索热词",
"carouselSearchHotWordInterval": "显示时间 / 秒",
"carouselSearchHotWordText": "内容",
"carouselSearchHotWordTextPlaceholder": "请输入热词",
"carouselSearchAddHotWordItem": "添加一个热词",
"carouselSearchLogoTips": "建议尺寸70px * 30px",
"carouselSearchTextTips": "搜索内容是默认展示数据,当添加搜索热词时,搜索内容隐藏; 当没有搜索热词时,搜索内容展示",
"carouselSearchPlaceholder": "请输入搜索内容",
"carouselSearchTabSet": "选项卡设置",
"carouselSearchTabControl": "展示开关",
"carouselSearchTabCategoryText": "分类名称",
"carouselSearchTabCategoryTextPlaceholder": "请输入分类名称",
"carouselSearchAddTabItem": "添加一个选项卡",
"selectSourcesDiyPage": "选择微页面",
"selectDiyPagePlaceholder": "请选择微页面",
"diyPageTypeName": "页面类型",
"diyPageForAddon": "所属应用",
"carouselSearchSwiperSet": "轮播图设置",
"carouselSearchSwiperControl": "展示开关",
"carouselSearchSwiperInterval": "切换间隔 / 秒",
"carouselSearchSwiperTips": "建议上传尺寸相同的图片推荐尺寸750*350鼠标拖拽可调整图片顺序",
"carouselSearchTabStyle": "选项卡样式",
"carouselSearchStyle": "搜索框样式",
"noColor": "常规颜色",
"selectColor": "选中颜色",
"fixedNoColor": "下滑常规颜色",
"fixedSelectColor": "下滑选中颜色",
"carouselSearchSwiperIndicatorSet": "指示器设置",
"carouselSearchSwiperIndicatorStyle": "指示器样式",
"carouselSearchSwiperStyle": "轮播样式",
"carouselSearchSwiperIndicatorStyle1": "样式1",
"carouselSearchSwiperIndicatorStyle2": "样式2",
"carouselSearchSwiperIndicatorStyle3": "样式3",
"carouselSearchSwiperIndicatorAlign": "显示位置",
"alignLeft": "居左",
"alignCenter": "居中",
"alignRight": "居右",
"horzLineStyle": "线条风格",
"horzLineStyleSolid": "实线",
"horzLineStyleDashed": "虚线",
"horzLineBorderColor": "线条颜色",
"horzLineBorderWidth": "线条宽度",
"floatBtnBtton": "按钮位置",
"floatBtnOffset": "上下偏移",
"floatBtnImageSet": "图片设置",
"floatBtnImageSize": "图片大小",
"floatBtnAroundRadius": "图片圆角",
"floatBtnImageSuggest": "建议上传正方形图片",
"topStatusBarImg": "图片",
"topStatusBarNav": "导航栏",
"topStatusBarNavTips": "此处控制当前页面导航栏是否显示",
"topStatusBarImgTips": "宽度自适应最大150px高度28px",
"topStatusBarTextColor": "标题颜色",
"topStatusBarBgColor": "头部颜色",
"rollTopStatusBarBgColor": "滚动后头部颜色",
"rollTopStatusBarTextColor": "滚动后标题颜色",
"topStatusBarSearchName": "搜索内容",
"topStatusBarSearchNamePlaceholder": "请输入搜索关键词",
"settingTips": "点击查看如何配置",
"pictureShowBlockOne": "模块一",
"pictureShowBlockTwo": "模块二",
"subTitleTextColor": "标题颜色",
"pictureShowBgColor": "背景颜色",
"pictureShowBtnText": "按钮文字",
"pictureShowBtnColor": "文字颜色",
"pictureShowBtnBgColor": "背景颜色",
"pictureShowBlockStyle": "模块样式",
"fieldNamePlaceholder": "请输入字段名称",
"fieldRemarkPlaceholder": "请输入字段说明",
"defaultValue": "默认值",
"defaultValuePlaceholder": "请输入默认值",
"formPlaceholder": "提示语",
"formPlaceholderTips": "请输入提示语",
"isRequired": "是否必填",
"optionPlaceholder": "请输入选项内容",
"formLayout": "表单布局",
"layoutStyle": "排版风格",
"singleTiling": "单列平铺‌",
"singleTilingTipsOne": "将所有需要填写的表单内容项直接罗列在页面上。",
"singleTilingTipsTwo": "适用于表单内容项较少且项目之间无逻辑关系的情况。",
"singleTilingTipsThree": "其优势在于相对简洁、便于操作",
"singleTilingTipsFour": "但当表单项数量较大时,一次性展示全部信息会增加用户的操作负担,填写效率较低",
"arrangeSideBySide": "左右排列‌",
"arrangeSideBySideTipsOne": "将表单分为左右两部分,左侧为标题和描述,右侧为输入区域。",
"arrangeSideBySideTipsTwo": "这种布局适用于标题和描述内容较少的情况,能够提高表单的紧凑性和用户体验。",
"layoutStyleTips": "切换后将同步所有表单组件的展示形式",
"borderControl": "边框开关",
"fieleContent": "字段内容",
"fieldName": "字段名称",
"filedRemark": "字段说明",
"otherSetting": "其他设置",
"hideControl": "隐藏该组件",
"hideControlTipsOne": "勾选后填表人填表时看不到该字段。",
"hideControlTipsTwo": "适用于你不再收集该字段又不希望删除已收集的数据。",
"textStyle": "文字样式",
"filedRemarkStyle": "字段说明样式",
"style": "样式",
"listStyle": "列表",
"dropDownStyle": "下拉",
"option": "选项",
"addSingleOption": "添加单个选项",
"addMultipleOption": "批量添加选项",
"addOptionTips": "每个选项之间用英文“,” 隔开,自动过滤重复内容",
"errorTipsOne": "存在重复选项,请检查内容",
"errorTipsTwo": "选项已存在,请重新输入",
"dataFormat": "日期格式",
"startDate": "开始日期",
"startTime": "开始时间",
"startDataTips": "开始日期不能为空",
"startTimeTips": "开始时间不能为空",
"startDataPlaceholder": "请选择开始日期",
"dataPlaceholder": "请选择日期",
"timePlaceholder": "请选择时间",
"startTimePlaceholder": "请选择开始时间",
"endDate": "结束日期",
"endTime": "结束时间",
"endDataPlaceholder": "请选择结束日期",
"endTimePlaceholder": "请选择结束时间",
"endDataTips": "结束日期不能为空",
"endTimeTips": "结束时间不能为空",
"startEndDataTips": "开始日期不能大于结束日期",
"startEndTimeTips": "开始时间不能大于结束时间",
"currentDate": "当天日期",
"diyDate": "指定日期",
"currentTime": "当天时间",
"diyTime": "指定时间",
"preventDuplication":"内容防重复",
"preventDuplicationTipsOne":"该组件填写的内容不能与已提交的数据重复。",
"preventDuplicationTipsTwo":"极端情况下可能存在延时导致限制失效。",
"privacyProtection":"隐私保护",
"privacyProtectionTipsOne":"会自动将提交的个人信息做加密展示。",
"privacyProtectionTipsTwo":"适用于公开展示收集的数据且不暴露用户隐私。",
"privacyProtectionTipsThree":"提交后自动隐藏中间11位数字仅管理员可查看",
"privacyProtectionTipsFour":"提交后自动隐藏文本,仅管理员可查看",
"privacyProtectionTipsFive":"提交后自动隐藏中间5位数字仅管理员可查看",
"imageLimit":"限制数量",
"imageLimitPlaceholder":"请输入限制数量",
"imageLimitErrorTips":"限制数量格式输入错误",
"imageLimitErrorTipsTwo":"限制数量不能小于0",
"imageLimitErrorTipsThree":"限制数量必须大于0",
"imafeLimitErrorTipsFour":"限制数量最大不能超过20",
"defaultValueTips":"设置后,默认值会自动填充到输入框,填表人可在此基础上进行修改。",
"defaultErrorTips":"默认值格式输入错误",
"defaultMustZeroTips":"默认值不能小于0",
"access":"获取方式",
"authorizeWeChatLocation":"授权微信定位",
"manuallySelectPositioning":"手动选择定位",
"unit":"单位",
"unitPlaceholder":"请输入单位",
"followContent":"跟随内容",
"hoverScreenBottom":"悬浮屏幕底部",
"btnTips":"当表单内容多时,只有滚动页面至最底部才会显示提交按钮",
"btnTipsTwo":"当表单内容多时,滚动页面至最底部时,提交按钮会自动按钮悬浮在屏幕底部,方便填表人快速提交显示在屏幕底部",
"btnTipsThree":"若前端以嵌入形式调用表单,提交按钮组件将不显示,相关业务由该页面负责处理",
"submitBtn":"提交按钮",
"submitBtnName":"按钮名称",
"btnNamePlaceholder":"请输入按钮名称",
"submitBtnNamePlaceholder":"请输入提交按钮名称",
"resetBtn":"重置按钮",
"btnStyle":"按钮样式",
"resetBtnNamePlaceholder":"请输入重置按钮名称",
"rowCount":"显示行数",
"rowCountPlaceholder":"请输入显示行数"
}

View File

@ -0,0 +1,126 @@
{
"title": "表单名称",
"typeName": "表单类型",
"forAddon": "所属应用",
"forAddonPlaceholder": "请选择所属应用",
"addFormTips": "创建新表单",
"formTypePlaceholder": "请选择表单类型",
"nameMax": "名称不能超过12个字符",
"status": "状态",
"updateTime": "更新时间",
"statusOn": "启用",
"statusOff": "禁用",
"all": "全部",
"wapUrl": "wap链接",
"weappUrl": "小程序链接",
"shareLink": "分享链接",
"copy": "复制",
"copySuccess": "复制成功",
"titlePlaceholder": "请输入表单名称",
"addDiyForm": "添加表单",
"diyFormDeleteTips": "确定要删除该表单吗?",
"diyFormCopyTips": "确定要复制该表单吗?",
"preview": "预览",
"share": "分享",
"shareSet": "分享设置",
"sharePage": "分享表单",
"wechat": "微信公众号",
"weapp": "微信小程序",
"shareTitle": "分享标题",
"shareTitlePlaceholder": "请输入分享标题",
"shareDesc": "分享描述",
"shareDescPlaceholder": "请输入分享描述",
"shareImageUrl": "分享图片",
"joinMemberType": "参与会员",
"allMember": "所有会员参与",
"selectedMemberLevel": "指定会员等级",
"selectedMemberLabel": "指定会员标签",
"memberLevel": "会员等级",
"memberLevelPlaceholder": "请选择会员等级",
"memberLabel": "会员标签",
"memberLabelPlaceholder": "请选择会员标签",
"labelTips": "请选择会员标签",
"levelTips": "请选择会员等级",
"batchDeletion": "批量删除",
"batchEmptySelectedFormsTips": "请选择要删除的表单",
"batchFormsDeleteTips": "确定要删除选中的表单吗?",
"promotion":"推广",
"submirSuccess":"提交成功页",
"writeSet":"填写设置",
"export":"导出",
"detail":"详情",
"more":"更多",
"formPromotion":"表单推广",
"promoteUrl":"推广链接",
"downLoadQRCode":"下载二维码",
"configureFailed":"配置失败",
"writeSuccess":"填写成功",
"viewFillingDetails":"查看填写详情",
"finish":"完成",
"finishTips":"点击后进入首页",
"back":"返回",
"backTips":"点击后返回表单",
"afterSubmission":"填表人提交后",
"displayTextMessages":"显示文字消息",
"displayTextMessagesTips":"提交后页面显示文字信息。",
"promptText":"提示文字",
"defaultPrompt":"默认提示",
"defaultPromptTips":"将显示: 填写成功",
"diyPrompt":"自定义提示",
"tipsTextPlaceholder":"感谢你的填写",
"subsequentPperationButtons":"后续操作按钮",
"validityPeriodOfVoucher":"凭证有效期",
"noLimit":"不限制",
"specifyTime":"设置固定有效期",
"specifyTimeTips":"每条记录的凭证有效期都是一样的,例如:会议凭证可设置有效期为会议举行时间。",
"submissionTime":"按提交时间设置有效期",
"submissionTimeTips":"每条记录的凭证有效期按照提交时间来计算例如优惠凭证的有效期可以设置为领取后3天内有效。",
"afterSubmissionRecords":"提交记录后",
"effective":"有效",
"voucherStyle":"凭证样式",
"titleAboveTheCode":"码上方标题",
"titleAboveTheCodePlaceholder":"请妥善保存你的核销凭证",
"contentAboveTheCode":"码上方内容",
"contentAboveTheCodePlaceholder":"请输入码上方内容",
"addLinefeeds":"添加换行符",
"addFields":"添加字段",
"contentBelowTheCode":"码下方内容",
"submissionRecordTime":"展示提交记录时间",
"currentTime":"展示当前时间",
"currentTimeTips":"会以秒进行跳动,可起到防作假的功能",
"voucherDeadline":"展示凭证截止时间",
"saveVoucher":"支持填表人保存凭证",
"dispalyPromptText":"展示提示文字",
"diy":"自定义",
"apieceFillQuantity":"每人可填写次数",
"fillQuantityTotal":"表单可填写总数",
"writeTips":"填写限制的校验在极端情况下可能存在延时,从而导致限制失效,不建议商品限时抢购等场景使用该功能",
"fillInTheTimePeriod":"可填写时间段",
"setSpecifyTime":"设置开始/停止时间",
"openDayTime":"设置每日开启时间",
"timeLimitRuleOne":"开始/停止时间不能为空",
"timeLimitRuleTwo":"开启时间不能为空",
"timeLimitRuleThree":"开始时间不能等于结束时间",
"numCannotNull":"次数不能为空",
"dataAndStatistics":"数据与统计",
"detailData":"明细数据",
"fillInFormPerson":"填表人",
"fillInFormPersonplaceholder":"请输入填表人",
"fillInFormDate":"填表时间",
"fillInFormPersonInfo":"填表人信息",
"fillInFormPersonStatics":"填表人统计",
"fillInFormTotal":"总计(表单填写数)",
"fieldStatistics":"字段统计",
"viewInformation":"查看信息",
"deleteTips":"确定删除该条数据吗"
}

View File

@ -46,5 +46,19 @@
"cashOutNumberPlaceholder": "请输入提现单号",
"alipayAccount": "支付宝账号",
"bankName": "银行名称",
"bankAccount": "银行卡号"
"bankAccount": "银行卡号",
"cashOutInfo":"收款方信息",
"transferCode":"收款码",
"realname":"真实姓名",
"account":"账号",
"bankRealname":"持卡人姓名",
"remark":"备注",
"remarkPlaceholder":"请输入备注",
"passAudit":"通过审核",
"transferVoucher":"转账凭证",
"transferVoucherPlaceholder":"请上传转账凭证",
"transferRemark":"转账补充说明",
"transferRemarkPlaceholder":"请输入转账补充说明",
"notes":"备注",
"check":"检查打款进度"
}

View File

@ -1,6 +1,5 @@
{
"siteTitle": "NIUCLOUD-ADMIN",
"siteDesc": "基于thinkphp6+elementplus+typescript等技术的多端saas通用管理框架采用restful的api接口设计前后端完全分离同时支持多语言开发。",
"logging": "登录中",
"platform": "管理端",
"login" : "登录",

View File

@ -5,7 +5,7 @@
"signPeriod": "签到周期",
"signPeriodTip": "请输入签到周期",
"signPeriodLimitTips": "签到周期格式错误",
"signPeriodMustZeroTips": "签到周期必须大于0",
"signPeriodMustZeroTips": "签到周期为2-365天",
"calendarSign": "日历签到",
"periodSign": "周期签到",
"daySignAward": "日签奖励",
@ -26,12 +26,16 @@
"ruleExplainTip": "请输入规则说明",
"ruleExplainDefault": "1.每日签到可以获得日签奖励,连续签到可以获得连签奖励;\n2.每日最多可签到1次断签则会重新计算连签天数\n3.活动以及奖励最终解释权归商家所有。",
"useDefaultExplain": "使用默认说明",
"continueSign": "连续签到天数",
"continueSign": "连签天数",
"continueSignFormatError": "连签天数格式错误",
"continueSignBerweenDays": "连签天数为2-365天",
"receiveLimit": "领取限制",
"noLimit": "不限制",
"everyOneLimit": "每人限领",
"time": "次",
"day": "天",
"continueSignPlaceholder":"请输入连续签到天数",
"receiveNumPlaceholder":"请输入限领次数"
"continueSignPlaceholder":"请输入连签天数",
"receiveNumPlaceholder":"请输入限领次数",
"receiveNumFormatError":"限领次数格式错误",
"receiveNumMustGreaterThanZeroTip":"限领次数不能小于等于0"
}

View File

@ -14,5 +14,8 @@
"automatedTransit": "自动转账",
"manualTransfer": "手动转账",
"wechat": "微信",
"alipay": "支付宝"
"alipay": "支付宝",
"minTips":"注意微信零钱最低提现金额为0.1",
"transferTips":"只有微信零钱支持自动转账,微信零钱可能会遇到资金不足、超过当日转账上限等因素的情况下会导致转账失败,停留在待转账状态下,需要管理员手动在后台操作",
"transferModeTips":"仅有微信零钱这一种转账方式支持线上打款,其余转账方式皆只支持线下打款"
}

View File

@ -81,5 +81,7 @@
"helpBtn":"帮付按钮名称",
"helpBtnPlaceholder":"请输入帮付按钮名称",
"remark":"发起帮付留言",
"remarkPlaceholder":"请输入留言备注"
"remarkPlaceholder":"请输入留言备注",
"payWechatImage":"默认分享图片(公众号)",
"payWeappImage":"默认分享图片(小程序)"
}

View File

@ -27,5 +27,7 @@
"appPublicCertPathTips": "上传appCertPublicKey文件",
"alipayPublicCertPathTips": "上传alipayCertPublicKey文件",
"alipayRootCertPathTips": "上传alipayRootCert文件",
"operationTip": "温馨提示:打款设置用于会员提现转账,发放红包等场景"
"operationTip": "温馨提示:打款设置用于会员提现转账,发放红包等场景",
"transferTips":"注意:应微信方规定,在2025年1月15日前开通商家转账到零钱服务的商户号可正常使用转账功能,之后开通的不支持使用转账到零钱服务"
}

View File

@ -23,7 +23,7 @@
"type":"插件类型",
"typePlaceholder":"请选择插件类型",
"typePlaceholder1":"应用指独立开发的系统比如商城零售erp等",
"typePlaceholder2":"插件:指不是独立的系统,可以是辅助应用的插件比如商城的拼团,也可以是独立的插件比如系统表单等",
"typePlaceholder2":"插件:指不是独立的系统,可以是辅助应用的插件比如商城的拼团,也可以是独立的插件比如万能表单等",
"supportType":"所属应用",
"supportApp":"支持应用",
"supportAppPlaceholder":"请选择支持应用",
@ -31,4 +31,4 @@
"successText":"检测当前插件标识尚未在应用市场注册插件开发后可以在niucloud官方市场发布",
"warningText":"检测到当前插件标识已经在niucloud官方市场注册开发的插件只能在本地使用无法在官方市场发布销售",
"onSaveSuccessText":"插件生成成功"
}
}

View File

@ -45,17 +45,14 @@
<el-button class="w-[154px] !h-[48px] mt-[8px]" type="primary" @click="authCodeApproveFn">授权码认证</el-button>
<el-popover ref="getAuthCodeDialog" placement="bottom-start" :width="478" trigger="click" class="mt-[8px]">
<div class="px-[18px] py-[8px]">
<p class="leading-[32px] text-[14px]">您在官方应用市场购买任意一款应用即可获得授权码输入正确授权码认证通过后即可支持在线升级和其它相关服务
</p>
<p class="leading-[32px] text-[14px]">您在官方应用市场购买任意一款应用即可获得授权码输入正确授权码认证通过后即可支持在线升级和其它相关服务</p>
<div class="flex justify-end mt-[36px]">
<el-button class="w-[182px] !h-[48px]" plain @click="market">去应用市场逛逛</el-button>
<el-button class="w-[100px] !h-[48px]" plain @click="getAuthCodeDialog.hide()">关闭</el-button>
</div>
</div>
<template #reference>
<el-button
class="w-[154px] !h-[48px] mt-[8px] !text-[var(--el-color-primary)] hover:!text-[var(--el-color-primary)] !bg-transparent"
plain type="primary">如何获取授权码?</el-button>
<el-button class="w-[154px] !h-[48px] mt-[8px] !text-[var(--el-color-primary)] hover:!text-[var(--el-color-primary)] !bg-transparent" plain type="primary">如何获取授权码?</el-button>
</template>
</el-popover>
</div>
@ -92,7 +89,7 @@
<div class="text-page-title mb-[20px]">历史版本</div>
<el-timeline>
<el-timeline-item :timestamp="item['release_time'] + ' 版本:' + item['version_no']" v-for="(item,index) in frameworkVersionList" type="primary" :hollow="true" placement="top" :key="index">
<div class="mt-[10px] p-[20px] bg-overlay rounded-md timeline-log-wrap whitespace-pre" v-if="item['upgrade_log']">
<div class="mt-[10px] p-[20px] bg-overlay rounded-md timeline-log-wrap whitespace-pre-wrap" v-if="item['upgrade_log']">
<div v-html="item['upgrade_log']"></div>
</div>
</el-timeline-item>
@ -108,9 +105,9 @@
import { reactive, ref, computed } from 'vue'
import { t } from '@/lang'
import { getVersions } from '@/app/api/auth'
import { getInstallConfig } from "@/app/api/sys"
import { getInstallConfig } from '@/app/api/sys'
import { getAuthInfo, setAuthInfo, getFrameworkVersionList } from '@/app/api/module'
import { ElMessageBox, FormInstance, FormRules,ElMessage } from 'element-plus'
import { ElMessageBox, FormInstance, FormRules, ElMessage } from 'element-plus'
import Upgrade from '@/app/components/upgrade/index.vue'
import CloudBuild from '@/app/components/cloud-build/index.vue'
import { cloneDeep } from 'lodash-es'

View File

@ -1,39 +1,44 @@
<template>
<div class="main-container w-full bg-white">
<!--应用管理-->
<div class="main-container" v-loading="loading">
<el-card class="box-card !border-none" shadow="never">
<div class="flex justify-between items-center">
<span class="text-page-title">应用管理</span>
</div>
<div class="flex flex-wrap plug-list pb-10 plug-large" v-if="appList.length">
<div v-for="(item, index) in appList" :key="index + 'b'" class="cursor-pointer mt-[20px] mr-4 bg-[#f7f7f7]" @click="toLink(item.key)">
<el-tooltip class="box-item" effect="light" placement="top">
<template #content>
<div class="max-w-[250px]">{{item.desc}}</div>
</template>
<div class="w-[264px] flex py-[20px] px-[17px] app-item relative">
<el-image class="w-[40px] h-[40px] mr-[10px]" :src="item.icon" fit="contain">
<template #error>
<div class="image-slot">
<img class="w-[40px] h-[40px]" src="@/app/assets/images/index/app_default.png" />
<template v-if="Object.keys(appList).length">
<template v-for="(item, index) in appList" :key="index + 'b'">
<template v-if="item.list.length">
<div class="flex justify-between items-center">
<span class="text-page-title">{{ item.title }}</span>
</div>
<div class="flex flex-wrap plug-list pb-10 plug-large">
<div class="cursor-pointer mt-[20px] mr-4 bg-[#f7f7f7]" v-for="(childItem,childIndex) in item.list" :key="childIndex" @click="toLink(childItem)">
<div class="w-[264px] flex py-[20px] px-[17px] app-item relative">
<el-image class="w-[40px] h-[40px] mr-[10px]" :src="img(childItem.icon)" fit="contain">
<template #error>
<div class="image-slot">
<img class="w-[40px] h-[40px]" src="@/app/assets/images/index/app_default.png" />
</div>
</template>
</el-image>
<div class="flex flex-col justify-between w-[180px]">
<div class="text-[14px] flex items-center">
<span class="app-text max-w-[170px]">{{ childItem.title }}</span>
<span class="iconfont iconxiaochengxu2 text-[#00b240] ml-[4px] !text-[14px]"></span>
</div>
<!-- <el-icon color="#666">
<QuestionFilled />
</el-icon> -->
<p class="app-text text-[12px] text-[#999]">{{childItem.desc}}</p>
</div>
</template>
</el-image>
<div class="flex flex-col justify-between w-[180px]">
<div class="text-[14px] flex items-center">
<span class="app-text max-w-[170px]">{{ item.title }}</span>
<span class="iconfont iconxiaochengxu2 text-[#00b240] ml-[4px] !text-[14px]"></span>
</div>
<!-- <el-icon color="#666">
<QuestionFilled />
</el-icon> -->
<p class="app-text text-[12px] text-[#999]">{{item.desc}}</p>
</div>
</div>
</el-tooltip>
</div>
</div>
<div class="empty flex items-center justify-center" v-if="!appList.length">
</template>
</template>
</template>
<div class="empty flex items-center justify-center" v-if="!loading && !Object.keys(appList).length">
<el-empty :description="t('emptyAppData')" />
</div>
</el-card>
@ -41,44 +46,50 @@
</template>
<script lang="ts" setup>
import { computed } from 'vue'
import { ref } from 'vue'
import { getShowApp } from '@/app/api/addon'
import { img } from '@/utils/common'
import useUserStore from '@/stores/modules/user'
import useSystemStore from '@/stores/modules/system'
import { useRouter } from 'vue-router'
import { t } from '@/lang'
const addonIndexRoute = useUserStore().addonIndexRoute
const router = useRouter()
const appList = computed(() => {
return useSystemStore().addons
})
const appList = ref<Record<string, any>[]>([])
const toLink = (addon: string) => {
addonIndexRoute[addon] && router.push({ name: addonIndexRoute[addon] })
const loading = ref(true)
const getAppList = async () => {
// const res = await getSiteAddons()
// appList.value = res.data
// loading.value = false
const res = await getShowApp()
appList.value = res.data
loading.value = false
}
getAppList()
const toLink = (item: any) => {
console.log('tol', item)
if (item.url) {
router.push(item.url)
} else {
addonIndexRoute[item.key] && router.push({ name: addonIndexRoute[item.key] })
}
}
</script>
<style lang="scss" scoped>
.main-container,
.empty {
min-height: calc(100vh - 84px);
}
.app-text {
overflow: hidden;
/* 超出部分隐藏 */
white-space: nowrap;
/* 禁止文本换行 */
text-overflow: ellipsis;
/* 显示省略号 */
}
.app-item:hover .with-ite {
display: block;
}
.el-form-item {
margin-bottom: 0px !important;
}
.app-text {
overflow: hidden;
/* 超出部分隐藏 */
white-space: nowrap;
/* 禁止文本换行 */
text-overflow: ellipsis;
/* 显示省略号 */
}
.app-item:hover{
transition: 0.5s;
box-shadow: 0px 2px 8px 0px rgba(0,0,0,0.1);
}
</style>

View File

@ -42,7 +42,7 @@
<el-form-item :label="t('authId')" prop="api_url" v-show="formData.menu_type != 0">
<el-input v-model.trim="formData.api_url" :placeholder="t('authIdPlaceholder')" class="input-width">
<template #append>
<el-select class="w-[90px] border-none" v-model="formData.methods">
<el-select class="border-none" style="width: 100px" v-model="formData.methods">
<el-option label="POST" value="post" />
<el-option label="GET" value="get" />
<el-option label="PUT" value="put" />
@ -77,7 +77,7 @@
</el-form-item>
<el-form-item :label="t('sort')">
<el-input-number v-model="formData.sort" :min="0" max="8" />
<el-input-number v-model="formData.sort" :min="0"/>
</el-form-item>
</el-form>

View File

@ -52,8 +52,7 @@ import { t } from '@/lang'
import type { FormInstance } from 'element-plus'
import { getUserInfo, addUser, editUser } from '@/app/api/user'
import { allRole } from '@/app/api/sys'
import { img, deepClone } from '@/utils/common'
import { AnyObject } from '@/types/global'
import { deepClone } from '@/utils/common'
const uid = ref<number | string>('')

View File

@ -53,13 +53,13 @@
<el-dialog v-model="dialogVisible" :title="t('codeDownTwoDesc')" width="30%" :before-close="handleClose">
<el-form ref="ruleFormRef" :model="form" label-width="120px">
<el-form-item prop="code" :label="t('code')">
<el-input v-model="form.code" :placeholder="t('codePlaceholder')" onkeyup="this.value = this.value.replace(/[^\d\.]/g,'');" />
<el-input v-model.trim="form.code" :placeholder="t('codePlaceholder')" onkeyup="this.value = this.value.replace(/[^\d\.]/g,'');" />
</el-form-item>
<el-form-item prop="path" :label="t('path')">
<upload-file v-model="form.path" :api="'weapp/upload'" :accept="'.zip'" />
</el-form-item>
<el-form-item :label="t('content')">
<el-input type="textarea" v-model="form.content" :placeholder="t('contentPlaceholder')" />
<el-input type="textarea" v-model.trim="form.content" :placeholder="t('contentPlaceholder')" />
</el-form-item>
</el-form>
<template #footer>
@ -73,7 +73,7 @@
</el-dialog>
<el-dialog v-model="failReasonDialogVisible" :title="t('failReason')" width="60%">
<el-scrollbar class="h-[60vh] w-full whitespace-pre p-[20px]">
<el-scrollbar class="h-[60vh] w-full whitespace-pre-wrap p-[20px]">
<div v-html="failReason"></div>
</el-scrollbar>
</el-dialog>

View File

@ -0,0 +1,119 @@
<template>
<el-dialog v-model="dialogThemeVisible" title="新增颜色" width="550px" align-center>
<el-form :model="formData" label-width="120px" ref="formRef" :rules="formRules">
<el-form-item label="名字" prop="title">
<el-input v-model="formData.title" class="!w-[250px]" maxlength="7" placeholder="请输入颜色名称" />
</el-form-item>
<el-form-item label="颜色key值" prop="label">
<el-input v-model="formData.label" class="!w-[250px]" maxlength="20" :disabled="type=='edit'" placeholder="请输入颜色key值" />
</el-form-item>
<el-form-item label="颜色value值" prop="value">
<el-color-picker v-model="formData.value" show-alpha :predefine="diyStore.predefineColors"/>
</el-form-item>
<el-form-item label="颜色提示">
<el-input v-model="formData.tip" class="!w-[250px]" placeholder="请输入颜色提示" />
</el-form-item>
</el-form>
<template #footer>
<div class="dialog-footer">
<el-button @click="dialogThemeVisible = false">取消</el-button>
<el-button type="primary" @click="confirmFn(formRef)">保存</el-button>
</div>
</template>
</el-dialog>
</template>
<script setup lang="ts">
import { ref, reactive, computed, watch } from 'vue'
import { t } from '@/lang'
import { getAreaListByPid } from '@/app/api/sys'
import { addMemberAddress } from '@/app/api/member'
import { filterNumber } from '@/utils/common'
import type { FormInstance } from 'element-plus'
import { FormRules } from 'element-plus'
import { cloneDeep } from 'lodash-es'
import useDiyStore from '@/stores/modules/diy'
const diyStore = useDiyStore()
const dialogThemeVisible = ref(false)
let confirmRepeat = false
const emit = defineEmits(['confirm'])
/**
* 表单数据
*/
const initialData = {
title: '',
label: '',
value: '',
tip: ''
}
let keyArr = []; // key
let type = ref('') //
const formData: Record<string, any> = reactive({ ...initialData })
const open = (option:any) => {
keyArr = option.key;
type.value = '';
//
for(let key in formData){
formData[key] = ''
}
if(option.data && Object.keys(option.data).length){
type.value = 'edit';
Object.keys(formData).forEach((item,index)=>{
formData[item] = option.data[item] ? option.data[item] : '';
})
}
dialogThemeVisible.value = true
}
const formRef = ref<FormInstance>()
//
const formRules = reactive<FormRules>({
title: [
{ required: true, message: "请输入颜色名称", trigger: 'blur' }
],
value: [
{ required: true, message: "请输入颜色value值", trigger: 'blur' }
],
label: [
{ required: true, message: "请输入颜色key值", trigger: 'blur' },
{
validator: (rule: any, value: any, callback: any) => {
const regex = /^[a-zA-Z0-9-]+$/
if (keyArr.indexOf(value) != -1) {
callback('新增颜色key值与已存在颜色key值命名重复请修改命名')
} if (!regex.test(value)) {
callback('颜色key值只能输入字母、数字和连字符')
} else{
callback();
}
},
trigger: 'blur'
}
]
})
const confirmFn = async (formEl: FormInstance | undefined) => {
if (confirmRepeat || !formEl) return
await formEl.validate(async (valid) => {
if (valid) {
emit('confirm', cloneDeep(formData));
dialogThemeVisible.value = false;
}
})
}
defineExpose({
dialogThemeVisible,
open
})
</script>
<style lang="scss" scoped>
</style>

View File

@ -4,7 +4,7 @@
<div class="edit-attr-item-wrap">
<h3 class="mb-[10px]">{{ t('titleContent') }}</h3>
<el-form label-width="80px" class="px-[10px]">
<el-form label-width="80px" class="px-[10px]" @submit.prevent>
<el-form-item :label="t('selectStyle')" class="flex">
<span class="text-primary flex-1 cursor-pointer" @click="showTitleStyle">{{ diyStore.editComponent.titleStyle.title }}</span>
<el-icon>
@ -97,7 +97,7 @@
<el-form-item :label="t('activeCubeSubTitle')" v-if="diyStore.editComponent.blockStyle.value != 'style-3'">
<el-input v-model.trim="item.subTitle.text" :placeholder="t('activeCubeSubTitlePlaceholder')" clearable maxlength="6" show-word-limit/>
</el-form-item>
<div v-show="diyStore.editComponent.blockStyle.value == 'style-4'">
<el-form-item :label="t('activeCubeSubTitleTextColor')">
<el-color-picker v-model="item.subTitle.textColor" show-alpha :predefine="diyStore.predefineColors" />
@ -108,13 +108,13 @@
<el-color-picker v-model="item.subTitle.endColor" show-alpha :predefine="diyStore.predefineColors"/>
</el-form-item>
</div>
<el-form-item :label="t('activeListFrameColor')">
<el-color-picker v-model="item.listFrame.startColor" show-alpha :predefine="diyStore.predefineColors" />
<icon name="iconfont iconmap-connect" size="20px" class="block !text-gray-400 mx-[5px]"/>
<el-color-picker v-model="item.listFrame.endColor" show-alpha :predefine="diyStore.predefineColors"/>
</el-form-item>
<div v-show="diyStore.editComponent.blockStyle.value != 'style-4' && diyStore.editComponent.blockStyle.value != 'style-3'">
<el-form-item :label="t('activeCubeButton')">
<el-input v-model.trim="item.moreTitle.text" :placeholder="t('activeCubeButtonPlaceholder')" clearable maxlength="3" show-word-limit/>
@ -427,7 +427,7 @@ const initBlockStyle = (style: any)=>{
diyStore.editComponent.blockStyle.fontWeight = "bold";
diyStore.editComponent.blockStyle.btnText = "italics";
diyStore.editComponent.list[0].title.textColor = "#303133";
diyStore.editComponent.list[0].subTitle.textColor = "#999999";
diyStore.editComponent.list[0].subTitle.startColor = "";
@ -505,7 +505,7 @@ const initBlockStyle = (style: any)=>{
}else if(style == 'style-4'){
diyStore.editComponent.blockStyle.fontWeight = "bold";
diyStore.editComponent.blockStyle.btnText = "normal";
diyStore.editComponent.list[0].title.textColor = "#303133";
diyStore.editComponent.list[0].subTitle.textColor = "#ED6E00";
diyStore.editComponent.list[0].subTitle.startColor = "#FFE4D9";

View File

@ -25,7 +25,7 @@
<div class="edit-attr-item-wrap">
<h3 class="mb-[10px]">{{ t('carouselSearchSet') }}</h3>
<el-form label-width="100px" class="px-[10px]">
<el-form label-width="100px" class="px-[10px]" @submit.prevent>
<el-form-item :label="t('selectStyle')" class="flex">
<span class="text-primary flex-1 cursor-pointer" @click="showSearchStyle">{{ diyStore.editComponent.search.styleName }}</span>
<el-icon>
@ -69,7 +69,7 @@
</el-dialog>
</div>
<div class="edit-attr-item-wrap mb-[20px]">
<h3 class="mb-[10px]">{{ t('carouselSearchHotWordSet') }}</h3>
<el-form label-width="100px" class="px-[10px]">
@ -97,11 +97,11 @@
</el-form>
</div>
<el-collapse v-model="activeNames" @change="handleChange" class="collapse-wrap">
<el-collapse-item :title="t('carouselSearchTabSet')" name="tab">
<div class="edit-attr-item-wrap">
<el-form label-width="100px" class="px-[10px]">
<el-form label-width="100px" class="px-[10px]" @submit.prevent>
<el-form-item :label="t('carouselSearchTabControl')">
<el-switch v-model="diyStore.editComponent.tab.control" />
</el-form-item>
@ -139,7 +139,7 @@
</div>
<!-- 选择微页面弹出框 -->
<el-dialog v-model="diyPageShowDialog" :title="t('selectSourcesDiyPage')" width="1000px" :close-on-press-escape="false" :destroy-on-close="true" :close-on-click-modal="false">
<el-dialog v-model="diyPageShowDialog" :title="t('selectSourcesDiyPage')" width="1000px" :close-on-press-escape="true" :destroy-on-close="true" :close-on-click-modal="false">
<el-table :data="diyPageTable.data" ref="diyPageTableRef" size="large" v-loading="diyPageTable.loading" height="490px" @current-change="handleCurrentDiyPageChange" row-key="id" highlight-current-row>
<template #empty>
<span>{{ !diyPageTable.loading ? t('emptyData') : '' }}</span>
@ -207,7 +207,7 @@
</el-form-item>
</el-form>
</div>
<div class="edit-attr-item-wrap" v-if="diyStore.editComponent.search.style == 'style-2'">
<h3 class="mb-[10px]">{{ t('carouselSearchSubTitleStyle') }}</h3>
<el-form label-width="100px" class="px-[10px]">

View File

@ -1,18 +1,23 @@
<template>
<!-- 内容 -->
<div class="content-wrap" v-show="diyStore.editTab == 'content'">
<div class="edit-attr-item-wrap">
<h3 class="mb-[10px]">{{ t('pageContent') }}</h3>
<el-form label-width="80px" class="px-[10px]">
<el-form label-width="80px" class="px-[10px]" @submit.prevent>
<el-form-item :label="t('diyPageTitle')">
<el-input v-model.trim="diyStore.pageTitle" :placeholder="t('diyPageTitlePlaceholder')" clearable maxlength="12" show-word-limit/>
<el-input v-model.trim="diyStore.pageTitle" :placeholder="t('diyPageTitlePlaceholder')" clearable maxlength="16" show-word-limit/>
<div class="text-sm text-gray-400">{{ t('pageTitleTips') }}</div>
</el-form-item>
</el-form>
</div>
<!-- 表单布局 页面设置 -->
<slot name="content"></slot>
<div class="edit-attr-item-wrap">
<h3 class="mb-[10px]">{{ t('statusBarContent') }}</h3>
<el-form label-width="80px" class="px-[10px]">
<el-form label-width="80px" class="px-[10px]" @submit.prevent>
<el-form-item :label="t('topStatusBarNav')" class="display-block">
<el-switch v-model="diyStore.global.topStatusBar.isShow"/>
<div class="text-sm text-gray-400">{{ t('statusBarSwitchTips') }}</div>

View File

@ -0,0 +1,258 @@
<template>
<el-dialog v-model="dialogThemeVisible" title="编辑色调" width="850px" align-center destroy-on-close="true">
<el-form :model="openData" label-width="150px" :rules="formRules">
<el-form-item label="色调名称" prop="title" >
<el-input v-model="openData.title" placeholder="请输入色调名称" maxlength="15" class="!w-[250px]" :disabled="openData.mark != 'diy'" />
</el-form-item>
</el-form>
<el-form :model="formData" label-width="150px" class="h-[640px] overflow-auto" ref="formRef" @submit.prevent>
<el-form-item :label="item.title" v-for="(item,index) in formData" :key="index" :prop="`${index}.value`" :rules="[{ required: true, message: `请选择${item.title}色调`, trigger: 'blur' }]">
<el-color-picker v-model="item.value" show-alpha :predefine="diyStore.predefineColors"/>
<div class="form-tip">{{item.tip}}</div>
</el-form-item>
<el-form-item :label="item.title" v-for="(item,index) in openData.diy_value" :key="index">
<div class="flex items-center">
<el-color-picker v-model="item.value" show-alpha :predefine="diyStore.predefineColors"/>
<span class="text-primary cursor-pointer text-[14px] ml-[20px]" @click="editThemeFn(item)">编辑</span>
<span class="text-primary cursor-pointer text-[14px] ml-[8px]" @click="deleteThemeFn(item)">删除</span>
</div>
<div class="form-tip">{{item.tip}}</div>
</el-form-item>
<el-form-item>
<div class="flex items-center text-primary cursor-pointer text-[14px]" @click="addThemeFn">
<span class="mr-[3px]">+</span>
<span>新增颜色</span>
</div>
<div class="form-tip">新增颜色key值不能与当前的存在的key值重复</div>
</el-form-item>
</el-form>
<add-theme ref="addThemeRef" @confirm="addThemeConfirm" />
<template #footer>
<div class="dialog-footer">
<el-button @click="dialogThemeVisible = false">取消</el-button>
<el-button type="primary" plain @click="resetConfirmFn()">重置</el-button>
<el-button type="primary" @click="confirmFn(formRef)">保存</el-button>
</div>
</template>
</el-dialog>
</template>
<script setup lang="ts">
import { ref, reactive, computed } from 'vue'
import { t } from '@/lang'
import { filterNumber } from '@/utils/common'
import { ElMessage } from 'element-plus'
import { cloneDeep } from 'lodash-es'
import addTheme from './add-theme.vue'
import useDiyStore from '@/stores/modules/diy'
import type { FormInstance } from 'element-plus'
const diyStore = useDiyStore()
const dialogThemeVisible = ref(false)
const addThemeRef = ref(null)
const openData: Record<string, any> = reactive({ //
title: '',
mark: '',
diy_value: [],
default: {},
data: {}
})
const emit = defineEmits(['confirm'])
const formRef = ref<FormInstance>()
//
const formRules = computed(() => {
return {
title: [
{ required: true, message: '请输入色调名称', trigger: 'blur' }
]
}
})
/**
* 表单数据
*/
const initialFormData = [
{
title: '主色调',
label: '--primary-color',
value: '#333333',
tip: '主色调在uiapp中使用var(--primary-color)'
},
{
title: '辅色调',
label: '--primary-help-color',
value: '#333333',
tip: '辅色调在uiapp中使用var(--primary-help-color)'
},
{
title: '页面背景色',
label: '--page-bg-color',
value: '#ffffff',
tip: '页面背景色在uiapp中使用var(--page-bg-color)'
},
{
title: '主色调浅色(淡)',
label: '--primary-color-light',
value: '',
tip: '主色调浅色在uiapp中使用var(--primary-color-light)'
},
{
title: '主色调浅色(深)',
label: '--primary-color-light2',
value: '',
tip: '主色调浅色在uiapp中使用var(--primary-color-light2)'
},
{
title: '灰色调',
label: '--primary-color-dark',
value: '#cccccc',
tip: '灰色调在uiapp中使用var(--primary-color-dark)'
},
{
title: '禁用色',
label: '--primary-color-disabled',
value: '#dddddd',
tip: '禁用色在uiapp中使用var(--primary-color-disabled)'
},
{
title: '价格颜色',
label: '--price-text-color',
value: '#333333',
tip: '价格颜色在uiapp中使用var(--price-text-color)'
}
]
const formData = ref([...cloneDeep(initialFormData)])
const open = (res:any) => { // name=>key=>default=>data=>
Object.keys(openData).forEach((key: string) => {
openData[key] = res[key] != undefined ? cloneDeep(res[key]) : ''
})
//
formData.value.forEach((item, index) => {
initialFormData.forEach((subItem, subIndex) => {
if (item.label == subItem.label) {
item.value = subItem.value
}
})
})
//
formData.value.forEach((item, index) => {
item.value = res.data[item.label] ? res.data[item.label] : item.value
})
console.log(formData.value, openData)
dialogThemeVisible.value = true
}
//
const addThemeFn = () => {
const keyArr = []
formData.value.forEach((item, index) => {
keyArr.push(item.label)
})
const obj = {
key: keyArr
}
addThemeRef.value.open(obj)
}
//
const editThemeFn = (res:any) => {
const keyArr = []
formData.value.forEach((item, index) => {
keyArr.push(item.label)
})
const obj = {
key: keyArr,
data: res
}
addThemeRef.value.open(obj)
}
//
const deleteThemeFn = (res:any) => {
let indent = -1
for (let i = 0; i < openData.diy_value.length; i++) {
if (openData.diy_value[i].label == res.label) {
indent = i
}
}
if (indent > -1) {
openData.diy_value.splice(indent, 1)
}
}
//
const addThemeConfirm = (res:any) =>{
for (let i = 0; i < openData.diy_value.length; i++) {
if (openData.diy_value[i].label == res.label) {
openData.diy_value[i] = res
return
}
}
openData.diy_value.push(res)
}
//
const resetConfirmFn = () => {
if (openData.default && Object.keys(openData.default).length) {
formData.value.forEach((item, index) => {
item.value = cloneDeep(openData.default[item.label])
})
} else {
formData.value = cloneDeep(initialFormData)
}
openData.diy_value = []
if (openData.mark == 'diy') {
openData.title = ''
}
ElMessage({
message: '重置成功',
type: 'success'
})
}
const confirmFn = async (formEl: FormInstance | undefined) => {
if (!formEl) return
await formEl.validate(async (valid) => {
if (valid) {
const params = {
theme: {},
diy_value: [],
title: ''
}
params.title = openData.title
formData.value.forEach((item, index) => {
params.theme[item.label] = item.value
})
openData.diy_value.forEach((item, index) => {
params.theme[item.label] = item.value
})
params.diy_value = openData.diy_value || []
emit('confirm', params)
dialogThemeVisible.value = false
}
})
}
defineExpose({
dialogThemeVisible,
open
})
</script>
<style lang="scss" scoped>
</style>

View File

@ -0,0 +1,174 @@
<template>
<el-dialog v-model="dialogThemeVisible" :title="data.addon_title" width="550px" align-center>
<el-form class="page-form mt-[15px]" :model="formData" label-width="90px" v-loading="loading">
<el-form-item label="选择配色">
<div class="flex items-center flex-wrap">
<template v-for="(tempItem,tempIndex) in theme_temp">
<div :key="tempIndex" v-if="tempItem.name != 'diy'" class="flex items-center border-[1px] border-solid border-[#dcdee2] rounded-[5px] h-[40px] px-[15px] mr-[10px] cursor-pointer my-[5px]" :class="{'!border-[var(--el-color-primary)]': curr_theme_mark == tempItem.name}" @click="themeTempChange(tempItem)">
<span v-if="data.theme" class="w-[20px] h-[20px] mr-[5px] rounded-[3px]" :style="{backgroundColor: data.theme['--primary-color']}"></span>
<span class="text-[14px]" :class="{'!text-[var(--el-color-primary)]': curr_theme_mark == tempItem.name}">{{tempItem.title}}</span>
</div>
</template>
<div class="flex items-center border-[1px] border-solid border-[#dcdee2] rounded-[5px] h-[40px] px-[15px] cursor-pointer" :class="{'!border-[var(--el-color-primary)]': curr_theme_mark == 'diy'}" @click="themeTempChange('diy')">
<span class="nc-iconfont nc-icon-tianjiaV6xx mr-[5px]" :class="{'!text-[var(--el-color-primary)]': curr_theme_mark == 'diy'}"></span>
<span class="text-[14px]" :class="{'!text-[var(--el-color-primary)]': curr_theme_mark == 'diy'}">自定义</span>
</div>
</div>
</el-form-item>
</el-form>
<edit-theme ref="editThemeRef" @confirm="editThemeConfirm"/>
<template #footer>
<div class="dialog-footer">
<el-button @click="dialogThemeVisible = false">取消</el-button>
<el-button type="primary" plain @click="editThemeFn()">编辑</el-button>
<el-button type="primary" @click="confirmFn()">确定</el-button>
</div>
</template>
</el-dialog>
</template>
<script setup lang="ts">
import { ref, reactive, computed, watch } from 'vue'
import { t } from '@/lang'
import { setDiyTheme, getDefaultTheme } from '@/app/api/diy'
import { cloneDeep } from 'lodash-es'
import editTheme from './edit-theme.vue'
import useDiyStore from '@/stores/modules/diy'
import { time } from 'echarts'
const diyStore = useDiyStore()
const editThemeRef = ref(null)
const dialogThemeVisible = ref(false)
let confirmRepeat = false
const curr_theme_title = ref('') //title
const curr_theme_mark = ref('') //
const curr_theme_value = ref('') //theme
const theme_temp = ref([]);
const mode = ref('default'); //
const data = ref({})
const open = (res:any) => {
confirmRepeat = false;
data.value = cloneDeep(res);
curr_theme_value.value = res.value;
curr_theme_mark.value = res.color_mark;
curr_theme_title.value = res.color_name;
//
theme_temp.value.forEach((item,index)=>{
if(item.name == data.value.color_mark){
item.diy_value = data.value.diy_value;
item.title = res.color_name;
}
})
mode.value = res.mode;
dialogThemeVisible.value = true
}
const emit = defineEmits(['confirm'])
const initData = () => {
getDefaultTheme().then((res) => {
theme_temp.value = res.data || [];
//
let diy_theme_temp = {
name: 'diy',
theme: '',
title: ''
}
theme_temp.value.push(diy_theme_temp);
})
}
initData()
//
const themeTempChange = (item)=>{
if(item.name == data.value.color_mark){ //
curr_theme_title.value = data.value.color_name;
curr_theme_mark.value = data.value.color_mark;
curr_theme_value.value = data.value.value;
}else if(typeof item == 'object'){ //
curr_theme_title.value = item.title;
curr_theme_mark.value = item.name;
curr_theme_value.value = item.theme;
}else{ //
curr_theme_title.value = '自定义';
curr_theme_mark.value = item;
curr_theme_value.value = '';
}
}
//
const editThemeFn = ()=>{
let theme = {
default: {}, //
data: {}, //
title:'',
mark: '', // ,
diy_value: [] //
}
theme.data = cloneDeep(curr_theme_value.value) || {};
theme.mark = curr_theme_mark.value;
theme_temp.value.forEach((item,index)=>{
if(item.name == curr_theme_mark.value){
theme.default = item.theme ? cloneDeep(item.theme) : '';
theme.diy_value= item.diy_value || [];
theme.title = item.title;
}
})
editThemeRef.value.open(theme)
}
//
const editThemeConfirm = (res)=>{
if(curr_theme_mark.value == data.value.color_mark){
data.value.value = res.theme;
}
theme_temp.value.forEach((item,index)=>{
if(item.name == curr_theme_mark.value){
item.diy_value= res.diy_value || [];
}
})
data.value.title = res.title;
curr_theme_value.value = res.theme;
}
//
const confirmFn = () => {
if (confirmRepeat) return
confirmRepeat = true
let params = {}
params.id = data.value.id;
params.mode = mode.value;
params.color_mark = curr_theme_mark.value;
params.value = curr_theme_value.value;
params.key = data.value.key;
params.color_name = curr_theme_mark.value == 'diy' ? (data.value.title || '自定义') : curr_theme_title.value;
theme_temp.value.forEach((item,index)=>{
if(item.name == curr_theme_mark.value){
params.diy_value = cloneDeep(item.diy_value);
}
})
setDiyTheme(params).then((res) => {
emit('confirm', data);
confirmRepeat = false;
dialogThemeVisible.value = false;
}).catch(()=>{
confirmRepeat = false;
})
}
defineExpose({
dialogThemeVisible,
open
})
</script>
<style lang="scss" scoped>
</style>

View File

@ -0,0 +1,95 @@
<template>
<div class="main-container">
<el-card class="box-card !border-none" shadow="never">
<div class="flex justify-between items-center">
<span class="text-page-title">{{ pageName }}</span>
</div>
<el-table :data="data" size="large" class="mt-[20px]" v-loading="loading">
<template #empty>
<span>{{ !loading ? t('emptyData') : '' }}</span>
</template>
<el-table-column label="应用" min-width="120" >
<template #default="{ row }">
<div class="flex items-center">
<el-image class="w-[40px] h-[40px] rounded-md overflow-hidden" :src="img(row.icon)" fit="contain">
<template #error>
<div class="flex items-center w-full h-full">
<img class="w-full h-full" src="@/app/assets/images/icon-addon.png" alt="">
</div>
</template>
</el-image>
<div class="flex-1 ml-2 truncate">{{ row.addon_title }}</div>
</div>
</template>
</el-table-column>
<el-table-column label="配色名称" min-width="120" >
<template #default="{ row }">
<div>{{ row.color_name }}</div>
</template>
</el-table-column>
<el-table-column label="配色方案" min-width="120" >
<template #default="{ row }">
<div class="rounded-[3px] inline-flex items-center justify-center border-[1px] border-solid border-[#f2f2f2] overflow-hidden" v-if="row.value">
<span class="w-[18px] h-[18px]" :style="{backgroundColor: row.value['--primary-color']}"></span>
<span class="w-[18px] h-[18px]" :style="{backgroundColor: row.value['--primary-help-color']}"></span>
<span class="w-[18px] h-[18px]" :style="{backgroundColor: '#fff'}"></span>
</div>
</template>
</el-table-column>
<el-table-column :label="t('operation')" align="right" fixed="right" width="100">
<template #default="{ row }">
<el-button type="primary" link @click="editEvent(row)">编辑</el-button>
</template>
</el-table-column>
</el-table>
</el-card>
<theme-list ref="themeListRef" @confirm="initData()" />
</div>
</template>
<script lang="ts" setup>
import { reactive, ref, watch, computed } from 'vue'
import { t } from '@/lang'
import { img } from '@/utils/common'
import { setDiyTheme, getDiyTheme, getDefaultTheme } from '@/app/api/diy'
import { useClipboard } from '@vueuse/core'
import { ElMessage, FormInstance } from 'element-plus'
import { ArrowLeft } from '@element-plus/icons-vue'
import { useRoute } from 'vue-router'
import themeList from './components/theme-list.vue'
import { cloneDeep } from 'lodash-es'
import { tr } from 'element-plus/es/locale'
const route = useRoute()
const pageName = route.meta.title
const loading = ref(true)
const themeListRef = ref(null)
const data = ref([])
const initData = () => {
loading.value = true;
getDiyTheme().then((res) => {
let obj = cloneDeep(res.data);
for(let key in obj){
obj[key].key = key;
}
data.value = Object.values(obj);
loading.value = false;
})
}
initData()
//
const editEvent = (data)=>{
themeListRef.value.open(data);
}
</script>
<style lang="scss" scoped></style>

View File

@ -0,0 +1,30 @@
<template>
<div>
<el-image v-for="(item,index) in props.data.handle_field_value" :src="img(item)" class="w-[70px] h-[70px]" :class="{ 'mr-[5px]' : (index + 1) < props.data.handle_field_value.length }" fit="contain" :preview-src-list="imgList" :zoom-rate="1.2" :max-scale="7"
:min-scale="0.2" :initial-index="index" :hide-on-click-modal="true" />
</div>
</template>
<script lang="ts" setup>
import { computed, reactive, ref } from 'vue'
import { img } from '@/utils/common'
import { t } from "@/lang";
const props = defineProps({
data: {
type: Object,
default: () => {
return {}
}
}
})
const imgList = computed(() => {
return props.data.handle_field_value.map((item: any) => {
return img(item)
})
})
</script>
<style lang="scss" scoped>
</style>

View File

@ -0,0 +1,25 @@
<template>
<div class="form-render">{{ props.data.render_value }}</div>
</template>
<script lang="ts" setup>
import { computed, reactive, ref } from 'vue'
import { t } from "@/lang";
const props = defineProps({
data: {
type: Object,
default: () => {
return {}
}
}
})
</script>
<style lang="scss" scoped>
.form-render {
word-wrap: break-word; /* 长单词或长字符串自动换行 */
word-break: break-word; /* 强制断词换行 */
white-space: pre-wrap; /* 保持空格和换行 */
}
</style>

View File

@ -0,0 +1,74 @@
<template>
<!-- 内容 -->
<div class="content-wrap" v-show="diyStore.editTab == 'content'">
<!-- 表单组件 字段内容设置 -->
<slot name="field"></slot>
<el-form label-width="100px" class="px-[10px]">
<el-form-item :label="t('地址格式')">
<el-radio-group v-model="diyStore.editComponent.addressFormat" class="!block">
<el-radio class="!block" label="province/city/district/address">{{ t('省/市/区/街道/详细地址') }}</el-radio>
<el-radio class="!block" label="province/city/district/street">{{ t('省/市/区/街道(镇)') }}</el-radio>
<el-radio class="!block" label="province/city/district">{{ t('省/市/区(县)') }}</el-radio>
<el-radio class="!block" label="province/city">{{ t('省/市') }}</el-radio>
<el-radio class="!block" label="province">{{ t('省') }}</el-radio>
</el-radio-group>
</el-form-item>
</el-form>
<!-- 表单组件 其他设置 -->
<slot name="other"></slot>
<el-form label-width="100px" class="px-[10px]">
<el-form-item class="display-block">
<template #label>
<div class="flex items-center">
<span class="mr-[3px]">{{ t('隐私保护') }}</span>
<el-tooltip effect="light" placement="top">
<template #content>
<p>会自动将提交的个人信息做加密展示</p>
<p>适用于公开展示收集的数据且不暴露用户隐私</p>
</template>
<el-icon>
<QuestionFilled color="#999999" />
</el-icon>
</el-tooltip>
</div>
</template>
<el-switch v-model="diyStore.editComponent.field.privacyProtection" />
<div class="text-sm text-gray-400">{{ t('提交后自动隐藏地址,仅管理员可查看') }}</div>
</el-form-item>
</el-form>
</div>
<!-- 样式 -->
<div class="style-wrap" v-show="diyStore.editTab == 'style'">
<!-- 表单组件 字段样式 -->
<slot name="style-field"></slot>
<!-- 组件样式 -->
<slot name="style"></slot>
</div>
</template>
<script lang="ts" setup>
import { t } from '@/lang'
import { ref } from 'vue'
import useDiyStore from '@/stores/modules/diy'
const diyStore = useDiyStore()
diyStore.editComponent.ignore = ['componentBgUrl'] //
//
diyStore.editComponent.verify = (index: number) => {
const res = { code: true, message: '' }
return res
}
defineExpose({})
</script>
<style lang="scss" scoped></style>

View File

@ -0,0 +1,194 @@
<template>
<!-- 内容 -->
<div class="content-wrap" v-show="diyStore.editTab == 'content'">
<!-- 表单组件 字段内容设置 -->
<slot name="field"></slot>
<el-form label-width="100px" class="px-[10px]">
<el-form-item :label="t('style')">
<el-radio-group v-model="diyStore.editComponent.style">
<el-radio label="style-1">{{ t('defaultSources') }}</el-radio>
<el-radio label="style-2">{{ t('listStyle') }}</el-radio>
<el-radio label="style-3">{{ t('dropDownStyle') }}</el-radio>
</el-radio-group>
</el-form-item>
<el-form-item :label="t('option')">
<div ref="formCheckboxRef">
<div v-for="(option, index) in diyStore.editComponent.options" :key="option.id" class="option-item flex items-center mb-[15px]">
<el-input v-model="diyStore.editComponent.options[index].text" class="!w-[215px]" :placeholder="t('optionPlaceholder')" maxlength="30" clearable />
<span v-if="diyStore.editComponent.options.length > 1" @click="removeOption(index)" class="cursor-pointer ml-[5px] nc-iconfont nc-icon-shanchu-yuangaizhiV6xx"></span>
</div>
</div>
<span class="text-primary cursor-pointer mr-[10px]" @click="addOption">{{ t('addSingleOption') }}</span>
<el-popover :visible="visible" placement="bottom" :width="300">
<p class="mb-[5px]">{{ t('addMultipleOption') }}</p>
<p class="text-[#888] text-[12px] mb-[5px]">{{ t('addOptionTips') }}</p>
<el-input v-model.trim="optionsValue" type="textarea" clearable maxlength="200" show-word-limit />
<div class="mt-[10px] text-right">
<el-button size="small" text @click="visible = false">{{ t('cancel') }}</el-button>
<el-button size="small" type="primary" @click="batchAddOptions">{{ t('confirm') }}</el-button>
</div>
<template #reference>
<span class="text-primary cursor-pointer" @click="visible = true">{{ t('addMultipleOption') }}</span>
</template>
</el-popover>
</el-form-item>
</el-form>
<!-- 表单组件 其他设置 -->
<slot name="other"></slot>
<!-- <el-form label-width="100px" class="px-[10px]">-->
<!-- <el-form-item class="display-block">-->
<!-- <template #label>-->
<!-- <div class="flex items-center">-->
<!-- <span class="mr-[3px]">{{ t('隐私保护') }}</span>-->
<!-- <el-tooltip effect="light" placement="top">-->
<!-- <template #content>-->
<!-- <p>会自动将提交的个人信息做加密展示</p>-->
<!-- <p>适用于公开展示收集的数据且不暴露用户隐私</p>-->
<!-- </template>-->
<!-- <el-icon>-->
<!-- <QuestionFilled color="#999999" />-->
<!-- </el-icon>-->
<!-- </el-tooltip>-->
<!-- </div>-->
<!-- </template>-->
<!-- <el-switch v-model="diyStore.editComponent.field.privacyProtection" />-->
<!-- <div class="text-sm text-gray-400">{{ t('提交后自动隐藏内容,仅管理员可查看') }}</div>-->
<!-- </el-form-item>-->
<!-- </el-form>-->
</div>
<!-- 样式 -->
<div class="style-wrap" v-show="diyStore.editTab == 'style'">
<!-- 表单组件 字段样式 -->
<slot name="style-field"></slot>
<!-- 组件样式 -->
<slot name="style"></slot>
</div>
</template>
<script lang="ts" setup>
import { t } from '@/lang'
import { ref, onMounted, nextTick} from 'vue'
import useDiyStore from '@/stores/modules/diy'
import Sortable from 'sortablejs'
import { range } from 'lodash-es'
import { ElMessage } from "element-plus";
const diyStore = useDiyStore()
diyStore.editComponent.ignore = ['componentBgUrl'] //
//
diyStore.editComponent.verify = (index: number) => {
const res = { code: true, message: '' }
let pass = true;
for (let i = 0; i < diyStore.value[index].options.length; i++) {
if (!diyStore.value[index].options[i].text) {
res.code = false;
res.message = t('optionPlaceholder');
pass = false;
break;
}
}
if (!pass) return res;
let uniqueOptions = uniqueByKey(diyStore.value[index].options, 'text')
if (uniqueOptions.length != diyStore.value[index].options.length) {
res.code = false;
res.message = t('errorTipsOne')
}
return res
}
diyStore.editComponent.options.forEach((item: any) => {
if (!item.id) item.id = diyStore.generateRandom()
})
const visible = ref(false)
const optionsValue = ref()
const addOption = () => {
diyStore.editComponent.options.push({
id: diyStore.generateRandom(),
text: '选项' + (diyStore.editComponent.options.length + 1)
});
};
const removeOption = (index:any) => {
diyStore.editComponent.options.splice(index, 1);
}
//
const batchAddOptions = () => {
if (optionsValue.value.trim()) {
const newOptions = optionsValue.value.split(',').map((option: any) => {
return {
id: diyStore.generateRandom(),
text: option.trim()
};
}).filter((option: any) => option.text !== '');
//
const uniqueNewOptions = uniqueByKey(newOptions, 'text');
//
const filteredNewOptions = uniqueNewOptions.filter((newOption: any) =>
!diyStore.editComponent.options.some((existingOption: any) => existingOption.text === newOption.text)
);
//
if (filteredNewOptions.length > 0) {
diyStore.editComponent.options.push(...filteredNewOptions);
} else {
ElMessage({
message: t('errorTipsTwo'),
type: "error",
});
}
optionsValue.value = '';
visible.value = false;
}
};
//
const uniqueByKey = (arr: any, key: any) => {
const seen = new Set();
return arr.filter((item: any) => {
const serializedKey = JSON.stringify(item[key]);
return seen.has(serializedKey) ? false : seen.add(serializedKey);
});
}
const formCheckboxRef = ref()
onMounted(() => {
nextTick(() => {
const sortable = Sortable.create(formCheckboxRef.value, {
group: 'option-item',
animation: 200,
onEnd: event => {
const temp = diyStore.editComponent.options[event.oldIndex!]
diyStore.editComponent.options.splice(event.oldIndex!, 1)
diyStore.editComponent.options.splice(event.newIndex!, 0, temp)
sortable.sort(
range(diyStore.editComponent.options.length).map(value => {
return value.toString()
})
)
}
})
})
})
defineExpose({})
</script>
<style lang="scss" scoped></style>

View File

@ -0,0 +1,206 @@
<template>
<!-- 内容 -->
<div class="content-wrap" v-show="diyStore.editTab == 'content'">
<!-- 表单组件 字段内容设置 -->
<slot name="field"></slot>
<el-form label-width="100px" class="px-[10px]">
<el-form-item :label="t('dataFormat')">
<el-radio-group v-model="diyStore.editComponent.dateFormat">
<div class="flex flex-col">
<el-radio label="YYYY年M月D日">{{ dateFormat.format1 }}</el-radio>
<el-radio label="YYYY-MM-DD">{{ dateFormat.format2 }}</el-radio>
<el-radio label="YYYY/MM/DD">{{ dateFormat.format3 }}</el-radio>
</div>
</el-radio-group>
</el-form-item>
</el-form>
<div class="edit-attr-item-wrap">
<h3 class="mb-[10px]">{{ t('startDate') }}</h3>
<el-form label-width="100px" class="px-[10px]" @submit.prevent>
<el-form-item :label="t('formPlaceholder')">
<el-input v-model.trim="diyStore.editComponent.start.placeholder" :placeholder="t('formPlaceholderTips')" clearable maxlength="15" show-word-limit />
</el-form-item>
<el-form-item :label="t('defaultValue')">
<el-switch v-model="diyStore.editComponent.start.defaultControl" />
</el-form-item>
<el-form-item v-if="diyStore.editComponent.start.defaultControl">
<el-radio-group v-model="diyStore.editComponent.start.dateWay">
<el-radio label="current">{{ t('currentDate') }}</el-radio>
<el-radio label="diy">{{ t('diyDate') }}</el-radio>
</el-radio-group>
</el-form-item>
<el-form-item v-if="diyStore.editComponent.start.defaultControl && diyStore.editComponent.start.dateWay == 'diy'">
<el-date-picker v-model="diyStore.editComponent.field.default.start.date" format="YYYY/MM/DD" value-format="YYYY-MM-DD" type="date" :placeholder="t('startDataPlaceholder')" @change="startDateChange" />
</el-form-item>
</el-form>
</div>
<div class="edit-attr-item-wrap">
<h3 class="mb-[10px]">{{ t('endDate') }}</h3>
<el-form label-width="100px" class="px-[10px]" @submit.prevent>
<el-form-item :label="t('formPlaceholder')">
<el-input v-model.trim="diyStore.editComponent.end.placeholder" :placeholder="t('formPlaceholderTips')" clearable maxlength="15" show-word-limit />
</el-form-item>
<el-form-item :label="t('defaultValue')">
<el-switch v-model="diyStore.editComponent.end.defaultControl" />
</el-form-item>
<el-form-item v-if="diyStore.editComponent.end.defaultControl">
<el-radio-group v-model="diyStore.editComponent.end.dateWay">
<el-radio label="current">{{ t('currentDate') }}</el-radio>
<el-radio label="diy">{{ t('diyDate') }}</el-radio>
</el-radio-group>
</el-form-item>
<el-form-item v-if="diyStore.editComponent.end.defaultControl && diyStore.editComponent.end.dateWay == 'diy'">
<el-date-picker :disabled-date="disabledEndDate" v-model="diyStore.editComponent.field.default.end.date" format="YYYY/MM/DD" value-format="YYYY-MM-DD" type="date" :placeholder="t('endDataPlaceholder')" @change="endDateChange" />
</el-form-item>
</el-form>
</div>
<!-- 表单组件 其他设置 -->
<slot name="other"></slot>
</div>
<!-- 样式 -->
<div class="style-wrap" v-show="diyStore.editTab == 'style'">
<div class="edit-attr-item-wrap">
<h3 class="mb-[10px]">{{ t('textStyle') }}</h3>
<el-form label-width="80px" class="px-[10px]">
<el-form-item :label="t('textFontSize')">
<el-slider v-model="diyStore.editComponent.fontSize" show-input size="small" class="ml-[10px] diy-nav-slider" :min="12" :max="18" />
</el-form-item>
<el-form-item :label="t('textFontWeight')">
<el-radio-group v-model="diyStore.editComponent.fontWeight">
<el-radio :label="'normal'">{{ t('fontWeightNormal') }}</el-radio>
<el-radio :label="'bold'">{{ t('fontWeightBold') }}</el-radio>
</el-radio-group>
</el-form-item>
<el-form-item :label="t('textColor')">
<el-color-picker v-model="diyStore.editComponent.textColor" />
</el-form-item>
</el-form>
</div>
<!-- 组件样式 -->
<slot name="style"></slot>
</div>
</template>
<script lang="ts" setup>
import { t } from '@/lang'
import { ref, reactive, onMounted } from 'vue'
import { timeTurnTimeStamp } from '@/utils/common'
import useDiyStore from '@/stores/modules/diy'
const diyStore = useDiyStore()
diyStore.editComponent.ignore = ['componentBgUrl'] //
//
diyStore.editComponent.verify = (index: number) => {
const res = { code: true, message: '' }
let starTime = diyStore.value[index].field.default.start.date
let endTime = diyStore.value[index].field.default.end.date
console.log(diyStore.value[index].field.default)
let today = new Date()
const hours = String(today.getHours()).padStart(2, '0')
const minutes = String(today.getMinutes()).padStart(2, '0')
if (diyStore.editComponent.start.dateWay == 'current') {
starTime = today.toISOString().split('T')[0]
}
if (diyStore.editComponent.end.dateWay == 'current') {
endTime = today.toISOString().split('T')[0]
}
if (diyStore.editComponent.start.defaultControl && starTime == '' && diyStore.editComponent.end.dateWay == 'diy') {
res.code = false
res.message = t('startDataTips')
return res
}
if (diyStore.editComponent.end.defaultControl && endTime == '' && diyStore.editComponent.end.dateWay == 'diy') {
res.code = false
res.message = t('endDataTips')
return res
}
if (diyStore.editComponent.start.defaultControl && diyStore.editComponent.end.defaultControl && timeTurnTimeStamp(starTime) > timeTurnTimeStamp(endTime)) {
res.code = false
res.message = t('startEndDataTips')
return res
}
return res
}
const dateFormat: any = reactive({
format1: '',
format2: '',
format3: ''
})
// -
const disabledEndDate = (time:Date)=>{
let cutoffDate = null
let bool = false
if(diyStore.editComponent.start && diyStore.editComponent.start.defaultControl){
if(diyStore.editComponent.start.dateWay == 'diy'){
cutoffDate = new Date(diyStore.editComponent.field.default.start.date)
}else{
cutoffDate = new Date()
}
bool = time.getTime() < cutoffDate.getTime()
}
return bool
}
onMounted(() => {
let today = new Date()
let endDate = new Date()
endDate.setDate(endDate.getDate() + 7) // 7
if (diyStore.editComponent.field.default.start.timestamp) {
diyStore.editComponent.field.default.start.date = today.toISOString().split('T')[0]
diyStore.editComponent.field.default.start.timestamp = parseInt(today.getTime() / 1000)
}
if (diyStore.editComponent.field.default.end.timestamp) {
diyStore.editComponent.field.default.end.date = endDate.toISOString().split('T')[0]
diyStore.editComponent.field.default.end.timestamp = parseInt(endDate.getTime() / 1000)
}
let year = today.getFullYear()
let month = String(today.getMonth() + 1).padStart(2, '0')
let day = String(today.getDate()).padStart(2, '0')
const hours = String(today.getHours()).padStart(2, '0')
const minutes = String(today.getMinutes()).padStart(2, '0')
dateFormat.format1 = `${year}${month}${day}`
dateFormat.format2 = `${year}-${month}-${day}`
dateFormat.format3 = `${year}/${month}/${day}`
dateFormat.format4 = `${year}-${month}-${day} ${hours}:${minutes}`
})
//
const startDateChange = (date)=>{
diyStore.editComponent.field.default.start.date = date
diyStore.editComponent.field.default.start.timestamp = timeTurnTimeStamp(date)
let endDate = new Date(date)
endDate.setDate(endDate.getDate() + 7)
diyStore.editComponent.field.default.end.date = endDate.toISOString().split('T')[0]
diyStore.editComponent.field.default.end.timestamp = parseInt(endDate.getTime() / 1000)
}
//
const endDateChange = (date)=>{
diyStore.editComponent.field.default.end.date = date
diyStore.editComponent.field.default.end.timestamp = timeTurnTimeStamp(date)
}
defineExpose({})
</script>
<style lang="scss" scoped></style>

View File

@ -0,0 +1,101 @@
<template>
<!-- 内容 -->
<div class="content-wrap" v-show="diyStore.editTab == 'content'">
<!-- 表单组件 字段内容设置 -->
<slot name="field"></slot>
<el-form label-width="100px" class="px-[10px]" @submit.prevent>
<el-form-item :label="t('dataFormat')">
<el-radio-group v-model="diyStore.editComponent.dateFormat" class="!block">
<el-radio class="!block" label="YYYY年M月D日">{{ dateFormat.format1 }}</el-radio>
<el-radio class="!block" label="YYYY-MM-DD">{{ dateFormat.format2 }}</el-radio>
<el-radio class="!block" label="YYYY/MM/DD">{{ dateFormat.format3 }}</el-radio>
<el-radio class="!block" label="YYYY-MM-DD HH:mm">{{ dateFormat.format4 }}</el-radio>
</el-radio-group>
</el-form-item>
<el-form-item :label="t('formPlaceholder')">
<el-input v-model.trim="diyStore.editComponent.placeholder" :placeholder="t('formPlaceholderTips')" clearable maxlength="15" show-word-limit />
</el-form-item>
<el-form-item :label="t('defaultValue')">
<el-switch v-model="diyStore.editComponent.defaultControl"/>
</el-form-item>
<el-form-item v-if="diyStore.editComponent.defaultControl">
<el-radio-group v-model="diyStore.editComponent.dateWay">
<el-radio label="current">{{ t('currentDate') }}</el-radio>
<el-radio label="diy">{{ t('diyDate') }}</el-radio>
</el-radio-group>
</el-form-item>
<el-form-item v-if="diyStore.editComponent.defaultControl && diyStore.editComponent.dateWay == 'diy'">
<el-date-picker v-if="diyStore.editComponent.dateFormat != 'YYYY-MM-DD HH:mm'" v-model="diyStore.editComponent.field.default.date" format="YYYY/MM/DD" value-format="YYYY-MM-DD" type="date" :placeholder="t('dataPlaceholder')" @change="dateChange"/>
<el-date-picker v-else v-model="diyStore.editComponent.field.default.date" format="YYYY/MM/DD HH:mm" value-format="YYYY-MM-DD HH:mm" type="datetime" :placeholder="t('dataPlaceholder')" @change="dateChange"/>
</el-form-item>
</el-form>
<!-- 表单组件 其他设置 -->
<slot name="other"></slot>
</div>
<!-- 样式 -->
<div class="style-wrap" v-show="diyStore.editTab == 'style'">
<!-- 表单组件 字段样式 -->
<slot name="style-field"></slot>
<!-- 组件样式 -->
<slot name="style"></slot>
</div>
</template>
<script lang="ts" setup>
import { t } from '@/lang'
import { ref,reactive,watch,onMounted } from 'vue'
import { timeTurnTimeStamp } from '@/utils/common'
import useDiyStore from '@/stores/modules/diy'
const diyStore = useDiyStore()
diyStore.editComponent.ignore = ['componentBgUrl'] //
//
diyStore.editComponent.verify = (index: number) => {
const res = { code: true, message: '' }
return res
}
const dateFormat: any = reactive({
format1: '',
format2: '',
format3: '',
format4: ''
});
onMounted(() => {
//
const today = new Date();
if (!diyStore.editComponent.field.default.date) {
diyStore.editComponent.field.default.date = today.toISOString().split('T')[0];
diyStore.editComponent.field.default.timestamp = today.getTime() / 1000;
}
let year = today.getFullYear();
let month = String(today.getMonth() + 1).padStart(2, '0');
let day = String(today.getDate()).padStart(2, '0');
const hours = String(today.getHours()).padStart(2, '0');
const minutes = String(today.getMinutes()).padStart(2, '0');
dateFormat.format1 = `${year}${month}${day}`;
dateFormat.format2 = `${year}-${month}-${day}`;
dateFormat.format3 = `${year}/${month}/${day}`;
dateFormat.format4 = `${year}-${month}-${day} ${hours}:${minutes}`;
});
const dateChange = (date: any)=>{
diyStore.editComponent.field.default.date = date;
diyStore.editComponent.field.default.timestamp = timeTurnTimeStamp(date);
}
defineExpose({})
</script>
<style lang="scss" scoped></style>

View File

@ -0,0 +1,48 @@
<template>
<!-- 内容 -->
<div class="content-wrap" v-show="diyStore.editTab == 'content'">
<!-- 表单组件 字段内容设置 -->
<slot name="field"></slot>
<el-form label-width="100px" class="px-[10px]" @submit.prevent>
<el-form-item :label="t('formPlaceholder')">
<el-input v-model.trim="diyStore.editComponent.placeholder" :placeholder="t('formPlaceholderTips')" clearable maxlength="15" show-word-limit />
</el-form-item>
</el-form>
<!-- 表单组件 其他设置 -->
<slot name="other"></slot>
</div>
<!-- 样式 -->
<div class="style-wrap" v-show="diyStore.editTab == 'style'">
<!-- 表单组件 字段样式 -->
<slot name="style-field"></slot>
<!-- 组件样式 -->
<slot name="style"></slot>
</div>
</template>
<script lang="ts" setup>
import { t } from '@/lang'
import { ref } from 'vue'
import useDiyStore from '@/stores/modules/diy'
const diyStore = useDiyStore()
diyStore.editComponent.ignore = ['componentBgUrl'] //
//
diyStore.editComponent.verify = (index: number) => {
const res = { code: true, message: '' }
// todo
return res
}
defineExpose({})
</script>
<style lang="scss" scoped></style>

View File

@ -0,0 +1,43 @@
<template>
<!-- 内容 -->
<div class="content-wrap" v-show="diyStore.editTab == 'content'">
<!-- 表单组件 字段内容设置 -->
<slot name="field"></slot>
<el-form label-width="100px" class="px-[10px]" @submit.prevent>
<el-form-item :label="t('限制上传大小')">
<el-input v-model.trim="diyStore.editComponent.limitUploadSize" clearable maxlength="15" />
/MBBit*1024
</el-form-item>
</el-form>
<!-- 表单组件 其他设置 -->
<slot name="other"></slot>
</div>
<!-- 样式 -->
<div class="style-wrap" v-show="diyStore.editTab == 'style'">
<!-- 表单组件 字段样式 -->
<slot name="style-field"></slot>
<!-- 组件样式 -->
<slot name="style"></slot>
</div>
</template>
<script lang="ts" setup>
import { t } from '@/lang'
import { ref } from 'vue'
import useDiyStore from '@/stores/modules/diy'
const diyStore = useDiyStore()
diyStore.editComponent.ignore = ['componentBgUrl'] //
defineExpose({})
</script>
<style lang="scss" scoped></style>

View File

@ -0,0 +1,85 @@
<template>
<!-- 内容 -->
<div class="content-wrap" v-show="diyStore.editTab == 'content'">
<!-- 表单组件 字段内容设置 -->
<slot name="field"></slot>
<el-form label-width="100px" class="px-[10px]" @submit.prevent>
<el-form-item :label="t('formPlaceholder')">
<el-input v-model.trim="diyStore.editComponent.placeholder" :placeholder="t('formPlaceholderTips')" clearable maxlength="15" show-word-limit />
</el-form-item>
</el-form>
<!-- 表单组件 其他设置 -->
<slot name="other"></slot>
<el-form label-width="100px" class="px-[10px]">
<el-form-item>
<template #label>
<div class="flex items-center">
<span class="mr-[3px]">{{ t('preventDuplication') }}</span>
<el-tooltip effect="light" placement="top">
<template #content>
<p>{{ t('preventDuplicationTipsOne') }}</p>
<p>{{ t('preventDuplicationTipsTwo') }}</p>
</template>
<el-icon>
<QuestionFilled color="#999999" />
</el-icon>
</el-tooltip>
</div>
</template>
<el-switch v-model="diyStore.editComponent.field.unique" />
</el-form-item>
<el-form-item class="display-block">
<template #label>
<div class="flex items-center">
<span class="mr-[3px]">{{ t('privacyProtection') }}</span>
<el-tooltip effect="light" placement="top">
<template #content>
<p>{{ t('privacyProtectionTipsOne') }}</p>
<p>{{ t('privacyProtectionTipsTwo') }}</p>
</template>
<el-icon>
<QuestionFilled color="#999999" />
</el-icon>
</el-tooltip>
</div>
</template>
<el-switch v-model="diyStore.editComponent.field.privacyProtection" />
<div class="text-sm text-gray-400">{{ t('privacyProtectionTipsThree') }}</div>
</el-form-item>
</el-form>
</div>
<!-- 样式 -->
<div class="style-wrap" v-show="diyStore.editTab == 'style'">
<!-- 表单组件 字段样式 -->
<slot name="style-field"></slot>
<!-- 组件样式 -->
<slot name="style"></slot>
</div>
</template>
<script lang="ts" setup>
import { t } from '@/lang'
import { ref } from 'vue'
import useDiyStore from '@/stores/modules/diy'
const diyStore = useDiyStore()
diyStore.editComponent.ignore = ['componentBgUrl'] //
//
diyStore.editComponent.verify = (index: number) => {
const res = { code: true, message: '' }
// todo
return res
}
defineExpose({})
</script>
<style lang="scss" scoped></style>

View File

@ -0,0 +1,88 @@
<template>
<!-- 内容 -->
<div class="content-wrap" v-show="diyStore.editTab == 'content'">
<!-- 表单组件 字段内容设置 -->
<slot name="field"></slot>
<el-form label-width="100px" class="px-[10px]" @submit.prevent>
<el-form-item :label="t('imageLimit')">
<el-input v-model.trim="diyStore.editComponent.limit" :placeholder="t('imageLimitPlaceholder')" clearable maxlength="2" />
</el-form-item>
<el-form-item :label="t('上传方式')">
<el-checkbox-group v-model="diyStore.editComponent.uploadMode" :min="1">
<el-checkbox label="拍照上传" value="take_pictures" />
<el-checkbox label="从相册选择" value="select_from_album" />
</el-checkbox-group>
</el-form-item>
</el-form>
<!-- 表单组件 其他设置 -->
<slot name="other"></slot>
</div>
<!-- 样式 -->
<div class="style-wrap" v-show="diyStore.editTab == 'style'">
<!-- 表单组件 字段样式 -->
<slot name="style-field"></slot>
<!-- 组件样式 -->
<slot name="style"></slot>
</div>
</template>
<script lang="ts" setup>
import { t } from '@/lang'
import { ref } from 'vue'
import useDiyStore from '@/stores/modules/diy'
const diyStore = useDiyStore()
diyStore.editComponent.ignore = ['componentBgUrl'] //
//
diyStore.editComponent.verify = (index: number) => {
const res = { code: true, message: '' }
if (diyStore.value[index].limit == '') {
res.code = false;
res.message = t('imageLimitPlaceholder')
return res;
}
if (isNaN(diyStore.value[index].limit) || !regExp.number.test(diyStore.value[index].limit)) {
res.code = false;
res.message = t('imageLimitErrorTips');
return res;
}
if (diyStore.value[index].limit < 0) {
res.code = false;
res.message = t('imageLimitErrorTipsTwo')
return res;
}
if (diyStore.value[index].limit == 0) {
res.code = false;
res.message = t('imageLimitErrorTipsThree')
return res;
}
if (diyStore.value[index].limit > 20) {
res.code = false;
res.message = t('imafeLimitErrorTipsFour')
return res;
}
return res
}
//
const regExp: any = {
required: /[\S]+/,
number: /^\d{0,10}$/,
digit: /^\d{0,10}(.?\d{0,2})$/,
special: /^\d{0,10}(.?\d{0,3})$/
}
defineExpose({})
</script>
<style lang="scss" scoped></style>

View File

@ -0,0 +1,99 @@
<template>
<!-- 内容 -->
<div class="content-wrap" v-show="diyStore.editTab == 'content'">
<!-- 表单组件 字段内容设置 -->
<slot name="field"></slot>
<el-form label-width="100px" class="px-[10px]" @submit.prevent>
<el-form-item :label="t('formPlaceholder')">
<el-input v-model.trim="diyStore.editComponent.placeholder" :placeholder="t('formPlaceholderTips')" clearable maxlength="15" show-word-limit />
</el-form-item>
<el-form-item>
<template #label>
<div class="flex items-center">
<span class="mr-[3px]">{{ t('defaultValue') }}</span>
<el-tooltip effect="light" :content="t('defaultValueTips')" placement="top">
<el-icon>
<QuestionFilled color="#999999" />
</el-icon>
</el-tooltip>
</div>
</template>
<el-input v-model.trim="diyStore.editComponent.field.default" :placeholder="t('defaultValuePlaceholder')" clearable maxlength="18" show-word-limit />
</el-form-item>
</el-form>
<!-- 表单组件 其他设置 -->
<slot name="other"></slot>
<el-form label-width="100px" class="px-[10px]">
<el-form-item>
<template #label>
<div class="flex items-center">
<span class="mr-[3px]">{{ t('preventDuplication') }}</span>
<el-tooltip effect="light" placement="top">
<template #content>
<p>{{ t('preventDuplicationTipsOne') }}</p>
<p>{{ t('preventDuplicationTipsTwo') }}</p>
</template>
<el-icon>
<QuestionFilled color="#999999" />
</el-icon>
</el-tooltip>
</div>
</template>
<el-switch v-model="diyStore.editComponent.field.unique" />
</el-form-item>
<!-- <el-form-item class="display-block">-->
<!-- <template #label>-->
<!-- <div class="flex items-center">-->
<!-- <span class="mr-[3px]">{{ t('隐私保护') }}</span>-->
<!-- <el-tooltip effect="light" placement="top">-->
<!-- <template #content>-->
<!-- <p>会自动将提交的个人信息做加密展示</p>-->
<!-- <p>适用于公开展示收集的数据且不暴露用户隐私</p>-->
<!-- </template>-->
<!-- <el-icon>-->
<!-- <QuestionFilled color="#999999" />-->
<!-- </el-icon>-->
<!-- </el-tooltip>-->
<!-- </div>-->
<!-- </template>-->
<!-- <el-switch v-model="diyStore.editComponent.field.privacyProtection" />-->
<!-- <div class="text-sm text-gray-400">{{ t('提交后自动隐藏文本,仅管理员可查看') }}</div>-->
<!-- </el-form-item>-->
</el-form>
</div>
<!-- 样式 -->
<div class="style-wrap" v-show="diyStore.editTab == 'style'">
<!-- 表单组件 字段样式 -->
<slot name="style-field"></slot>
<!-- 组件样式 -->
<slot name="style"></slot>
</div>
</template>
<script lang="ts" setup>
import { t } from '@/lang'
import { ref } from 'vue'
import useDiyStore from '@/stores/modules/diy'
const diyStore = useDiyStore()
diyStore.editComponent.ignore = ['componentBgUrl'] //
//
diyStore.editComponent.verify = (index: number) => {
const res = { code: true, message: '' }
// todo
return res
}
defineExpose({})
</script>
<style lang="scss" scoped></style>

View File

@ -0,0 +1,71 @@
<template>
<!-- 内容 -->
<div class="content-wrap" v-show="diyStore.editTab == 'content'">
<!-- 表单组件 字段内容设置 -->
<slot name="field"></slot>
<el-form label-width="100px" class="px-[10px]">
<el-form-item :label="t('access')">
<el-radio-group v-model="diyStore.editComponent.mode">
<el-radio class="!mr-[20px]" label="authorized_wechat_location">{{ t('authorizeWeChatLocation') }}</el-radio>
<el-radio label="open_choose_location">{{ t('manuallySelectPositioning') }}</el-radio>
</el-radio-group>
</el-form-item>
</el-form>
<!-- 表单组件 其他设置 -->
<slot name="other"></slot>
<el-form label-width="100px" class="px-[10px]">
<el-form-item class="display-block">
<template #label>
<div class="flex items-center">
<span class="mr-[3px]">{{ t('privacyProtection') }}</span>
<el-tooltip effect="light" placement="top">
<template #content>
<p>{{ t('privacyProtectionTipsOne') }}</p>
<p>{{ t('privacyProtectionTipsTwo') }}</p>
</template>
<el-icon>
<QuestionFilled color="#999999" />
</el-icon>
</el-tooltip>
</div>
</template>
<el-switch v-model="diyStore.editComponent.field.privacyProtection" />
<div class="text-sm text-gray-400">{{ t('privacyProtectionTipsFour') }}</div>
</el-form-item>
</el-form>
</div>
<!-- 样式 -->
<div class="style-wrap" v-show="diyStore.editTab == 'style'">
<!-- 表单组件 字段样式 -->
<slot name="style-field"></slot>
<!-- 组件样式 -->
<slot name="style"></slot>
</div>
</template>
<script lang="ts" setup>
import { t } from '@/lang'
import useDiyStore from '@/stores/modules/diy'
import { ref } from 'vue'
const diyStore = useDiyStore()
diyStore.editComponent.ignore = ['componentBgUrl'] //
//
diyStore.editComponent.verify = (index: number) => {
const res = { code: true, message: '' }
// todo
return res
}
defineExpose({})
</script>
<style lang="scss" scoped></style>

View File

@ -0,0 +1,85 @@
<template>
<!-- 内容 -->
<div class="content-wrap" v-show="diyStore.editTab == 'content'">
<!-- 表单组件 字段内容设置 -->
<slot name="field"></slot>
<el-form label-width="100px" class="px-[10px]" @submit.prevent>
<el-form-item :label="t('formPlaceholder')">
<el-input v-model.trim="diyStore.editComponent.placeholder" :placeholder="t('formPlaceholderTips')" clearable maxlength="15" show-word-limit />
</el-form-item>
</el-form>
<!-- 表单组件 其他设置 -->
<slot name="other"></slot>
<el-form label-width="100px" class="px-[10px]">
<el-form-item>
<template #label>
<div class="flex items-center">
<span class="mr-[3px]">{{ t('preventDuplication') }}</span>
<el-tooltip effect="light" placement="top">
<template #content>
<p>{{ t('preventDuplicationTipsOne') }}</p>
<p>{{ t('preventDuplicationTipsTwo') }}</p>
</template>
<el-icon>
<QuestionFilled color="#999999" />
</el-icon>
</el-tooltip>
</div>
</template>
<el-switch v-model="diyStore.editComponent.field.unique" />
</el-form-item>
<el-form-item class="display-block">
<template #label>
<div class="flex items-center">
<span class="mr-[3px]">{{ t('privacyProtection') }}</span>
<el-tooltip effect="light" placement="top">
<template #content>
<p>{{ t('privacyProtectionTipsOne') }}</p>
<p>{{ t('privacyProtectionTipsTwo') }}</p>
</template>
<el-icon>
<QuestionFilled color="#999999" />
</el-icon>
</el-tooltip>
</div>
</template>
<el-switch v-model="diyStore.editComponent.field.privacyProtection" />
<div class="text-sm text-gray-400">{{ t('privacyProtectionTipsFive') }}</div>
</el-form-item>
</el-form>
</div>
<!-- 样式 -->
<div class="style-wrap" v-show="diyStore.editTab == 'style'">
<!-- 表单组件 字段样式 -->
<slot name="style-field"></slot>
<!-- 组件样式 -->
<slot name="style"></slot>
</div>
</template>
<script lang="ts" setup>
import { t } from '@/lang'
import useDiyStore from '@/stores/modules/diy'
import { ref } from 'vue'
const diyStore = useDiyStore()
diyStore.editComponent.ignore = ['componentBgUrl'] //
//
diyStore.editComponent.verify = (index: number) => {
const res = { code: true, message: '' }
// todo
return res
}
defineExpose({})
</script>
<style lang="scss" scoped></style>

View File

@ -0,0 +1,119 @@
<template>
<!-- 内容 -->
<div class="content-wrap" v-show="diyStore.editTab == 'content'">
<!-- 表单组件 字段内容设置 -->
<slot name="field"></slot>
<el-form label-width="100px" class="px-[10px]" @submit.prevent>
<el-form-item :label="t('formPlaceholder')">
<el-input v-model.trim="diyStore.editComponent.placeholder" :placeholder="t('formPlaceholderTips')" clearable maxlength="15" show-word-limit />
</el-form-item>
<el-form-item :label="t('unit')">
<el-input v-model.trim="diyStore.editComponent.unit" :placeholder="t('unitPlaceholder')" clearable maxlength="5" show-word-limit />
</el-form-item>
<el-form-item>
<template #label>
<div class="flex items-center">
<span class="mr-[3px]">{{ t('defaultValue') }}</span>
<el-tooltip effect="light" :content="t('defaultValueTips')" placement="top">
<el-icon>
<QuestionFilled color="#999999" />
</el-icon>
</el-tooltip>
</div>
</template>
<el-input v-model.trim="diyStore.editComponent.field.default" :placeholder="t('defaultValuePlaceholder')" @keyup="filterDigit($event)" clearable maxlength="18" show-word-limit />
</el-form-item>
</el-form>
<!-- 表单组件 其他设置 -->
<slot name="other"></slot>
<el-form label-width="100px" class="px-[10px]">
<!-- <el-form-item>-->
<!-- <template #label>-->
<!-- <div class="flex items-center">-->
<!-- <span class="mr-[3px]">{{ t('内容防重复') }}</span>-->
<!-- <el-tooltip effect="light" placement="top">-->
<!-- <template #content>-->
<!-- <p>该组件填写的内容不能与已提交的数据重复</p>-->
<!-- <p>极端情况下可能存在延时导致限制失效</p>-->
<!-- </template>-->
<!-- <el-icon>-->
<!-- <QuestionFilled color="#999999" />-->
<!-- </el-icon>-->
<!-- </el-tooltip>-->
<!-- </div>-->
<!-- </template>-->
<!-- <el-switch v-model="diyStore.editComponent.field.unique" />-->
<!-- </el-form-item>-->
<!-- <el-form-item class="display-block">-->
<!-- <template #label>-->
<!-- <div class="flex items-center">-->
<!-- <span class="mr-[3px]">{{ t('隐私保护') }}</span>-->
<!-- <el-tooltip effect="light" placement="top">-->
<!-- <template #content>-->
<!-- <p>会自动将提交的个人信息做加密展示</p>-->
<!-- <p>适用于公开展示收集的数据且不暴露用户隐私</p>-->
<!-- </template>-->
<!-- <el-icon>-->
<!-- <QuestionFilled color="#999999" />-->
<!-- </el-icon>-->
<!-- </el-tooltip>-->
<!-- </div>-->
<!-- </template>-->
<!-- <el-switch v-model="diyStore.editComponent.field.privacyProtection" />-->
<!-- <div class="text-sm text-gray-400">{{ t('提交后自动隐藏数字,仅管理员可查看') }}</div>-->
<!-- </el-form-item>-->
</el-form>
</div>
<!-- 样式 -->
<div class="style-wrap" v-show="diyStore.editTab == 'style'">
<!-- 表单组件 字段样式 -->
<slot name="style-field"></slot>
<!-- 组件样式 -->
<slot name="style"></slot>
</div>
</template>
<script lang="ts" setup>
import { t } from '@/lang'
import { ref } from 'vue'
import useDiyStore from '@/stores/modules/diy'
import { filterDigit } from '@/utils/common'
const diyStore = useDiyStore()
diyStore.editComponent.ignore = ['componentBgUrl'] //
//
diyStore.editComponent.verify = (index: number) => {
const res = { code: true, message: '' }
if(diyStore.value[index].field.default){
if (isNaN(diyStore.value[index].field.default) || !regExp.digit.test(diyStore.value[index].field.default)) {
res.code = false;
res.message = t('defaultErrorTips');
} else if (diyStore.value[index].field.default < 0) {
res.code = false;
res.message = t('defaultMustZeroTips')
}
}
return res
}
//
const regExp: any = {
required: /[\S]+/,
number: /^\d{0,10}$/,
digit: /^\d{0,10}(.?\d{0,2})$/,
special: /^\d{0,10}(.?\d{0,3})$/
}
defineExpose({})
</script>
<style lang="scss" scoped></style>

View File

@ -0,0 +1,214 @@
<template>
<!-- 内容 -->
<div class="content-wrap" v-show="diyStore.editTab == 'content'">
<!-- 表单组件 字段内容设置 -->
<slot name="field"></slot>
<el-form label-width="100px" class="px-[10px]">
<el-form-item :label="t('style')">
<el-radio-group v-model="diyStore.editComponent.style">
<el-radio label="style-1">{{ t('defaultSources') }}</el-radio>
<el-radio label="style-2">{{ t('listStyle') }}</el-radio>
<el-radio label="style-3">{{ t('dropDownStyle') }}</el-radio>
</el-radio-group>
</el-form-item>
<el-form-item :label="t('option')">
<div ref="formRadioRef">
<div v-for="(option, index) in diyStore.editComponent.options" :key="option.id" class="option-item flex items-center mb-[15px]">
<el-input v-model="diyStore.editComponent.options[index].text" class="!w-[215px]" :placeholder="t('optionPlaceholder')" clearable maxlength="30" />
<span v-if="diyStore.editComponent.options.length > 1" @click="removeOption(index)" class="cursor-pointer ml-[5px] nc-iconfont nc-icon-shanchu-yuangaizhiV6xx"></span>
</div>
</div>
<span class="text-primary cursor-pointer mr-[10px]" @click="addOption">{{ t('addSingleOption') }}</span>
<el-popover :visible="visible" placement="bottom" :width="300">
<p class="mb-[5px]">{{ t('addMultipleOption') }}</p>
<p class="text-[#888] text-[12px] mb-[5px]">{{ t('addOptionTips') }}</p>
<el-input v-model.trim="optionsValue" type="textarea" clearable maxlength="200" show-word-limit />
<div class="mt-[10px] text-right">
<el-button size="small" text @click="visible = false">{{ t('cancel') }}</el-button>
<el-button size="small" type="primary" @click="batchAddOptions">{{ t('confirm') }}</el-button>
</div>
<template #reference>
<span class="text-primary cursor-pointer" @click="visible = true">{{ t('addMultipleOption') }}</span>
</template>
</el-popover>
</el-form-item>
<!-- <el-form-item class="display-block">
<template #label>
<div class="flex items-center">
<span class="mr-[3px]">{{ t('逻辑规则') }}</span>
<el-tooltip effect="light" placement="top">
<template #content>
<p>支持选择某个选项后显示特定的组件</p>
</template>
<el-icon>
<QuestionFilled color="#999999" />
</el-icon>
</el-tooltip>
</div>
</template>
<div>
<el-button plain>{{ t('添加字段显示规则') }}</el-button>
<span class="mr-[3px]">1条字段显示规则</span>
<span class="text-primary cursor-pointer" @click="">设置</span>
</div>
</el-form-item> -->
</el-form>
<!-- 表单组件 其他设置 -->
<slot name="other"></slot>
<!-- <el-form label-width="100px" class="px-[10px]">-->
<!-- <el-form-item class="display-block">-->
<!-- <template #label>-->
<!-- <div class="flex items-center">-->
<!-- <span class="mr-[3px]">{{ t('隐私保护') }}</span>-->
<!-- <el-tooltip effect="light" placement="top">-->
<!-- <template #content>-->
<!-- <p>会自动将提交的个人信息做加密展示</p>-->
<!-- <p>适用于公开展示收集的数据且不暴露用户隐私</p>-->
<!-- </template>-->
<!-- <el-icon>-->
<!-- <QuestionFilled color="#999999" />-->
<!-- </el-icon>-->
<!-- </el-tooltip>-->
<!-- </div>-->
<!-- </template>-->
<!-- <el-switch v-model="diyStore.editComponent.field.privacyProtection" />-->
<!-- <div class="text-sm text-gray-400">{{ t('提交后自动隐藏内容,仅管理员可查看') }}</div>-->
<!-- </el-form-item>-->
<!-- </el-form>-->
</div>
<!-- 样式 -->
<div class="style-wrap" v-show="diyStore.editTab == 'style'">
<!-- 表单组件 字段样式 -->
<slot name="style-field"></slot>
<!-- 组件样式 -->
<slot name="style"></slot>
</div>
</template>
<script lang="ts" setup>
import { t } from '@/lang'
import { ref, onMounted, nextTick} from 'vue'
import useDiyStore from '@/stores/modules/diy'
import Sortable from 'sortablejs'
import { range } from 'lodash-es'
import { FormInstance, ElMessage } from "element-plus";
const diyStore = useDiyStore()
diyStore.editComponent.ignore = ['componentBgUrl'] //
//
diyStore.editComponent.verify = (index: number) => {
const res = { code: true, message: '' }
let pass = true;
for (let i = 0; i < diyStore.value[index].options.length; i++) {
if (!diyStore.value[index].options[i].text) {
res.code = false;
res.message = t('optionPlaceholder');
pass = false;
break;
}
}
if (!pass) return res;
let uniqueOptions = uniqueByKey(diyStore.value[index].options, 'text')
if (uniqueOptions.length != diyStore.value[index].options.length) {
res.code = false;
res.message = t('errorTipsOne')
}
return res
}
diyStore.editComponent.options.forEach((item: any) => {
if (!item.id) item.id = diyStore.generateRandom()
})
const visible = ref(false)
const optionsValue = ref()
const addOption = () => {
diyStore.editComponent.options.push({
id: diyStore.generateRandom(),
text: '选项' + (diyStore.editComponent.options.length + 1)
});
};
const removeOption = (index:any) => {
diyStore.editComponent.options.splice(index, 1);
}
const batchAddOptions = () => {
if (optionsValue.value.trim()) {
const newOptions = optionsValue.value.split(',').map((option: any) => {
return {
id: diyStore.generateRandom(),
text: option.trim()
};
}).filter((option: any) => option.text !== '');
//
const uniqueNewOptions = uniqueByKey(newOptions, 'text');
//
const filteredNewOptions = uniqueNewOptions.filter(newOption =>
!diyStore.editComponent.options.some(existingOption => existingOption.text === newOption.text)
);
//
if (filteredNewOptions.length > 0) {
diyStore.editComponent.options.push(...filteredNewOptions);
} else {
ElMessage({
message: t('errorTipsTwo'),
type: "warning",
});
}
optionsValue.value = '';
visible.value = false;
}
};
//
const uniqueByKey = (arr: any, key: any) => {
const seen = new Set();
return arr.filter((item: any) => {
const serializedKey = JSON.stringify(item[key]);
return seen.has(serializedKey) ? false : seen.add(serializedKey);
});
}
const formRadioRef = ref()
onMounted(() => {
nextTick(() => {
const sortable = Sortable.create(formRadioRef.value, {
group: 'option-item',
animation: 200,
onEnd: event => {
const temp = diyStore.editComponent.options[event.oldIndex!]
diyStore.editComponent.options.splice(event.oldIndex!, 1)
diyStore.editComponent.options.splice(event.newIndex!, 0, temp)
sortable.sort(
range(diyStore.editComponent.options.length).map(value => {
return value.toString()
})
)
}
})
})
})
defineExpose({})
</script>
<style lang="scss" scoped></style>

View File

@ -0,0 +1,113 @@
<template>
<!-- 内容 -->
<div class="content-wrap" v-show="diyStore.editTab == 'content'">
<div class="edit-attr-item-wrap">
<el-form label-width="80px" class="px-[10px]">
<el-form-item :label="t('floatBtnBtton')" class="display-block">
<el-radio-group v-model="diyStore.editComponent.btnPosition" @change="btnPositionChangeFn">
<el-radio label="follow_content">{{ t('followContent') }}</el-radio>
<el-radio label="hover_screen_bottom">{{ t('hoverScreenBottom') }}</el-radio>
</el-radio-group>
<div class="text-sm text-gray-400 mb-[5px] leading-[1.4]" v-show="diyStore.editComponent.btnPosition == 'follow_content'">{{ t('btnTips') }}</div>
<div class="text-sm text-gray-400 mb-[5px]" v-show="diyStore.editComponent.btnPosition == 'hover_screen_bottom'">{{ t('btnTipsTwo') }}</div>
<div class="text-sm text-gray-400 mb-[10px] leading-[1.4]">{{ t('btnTipsThree') }}</div>
</el-form-item>
</el-form>
</div>
<div class="edit-attr-item-wrap">
<h3 class="mb-[10px]">{{ t('submitBtn') }}</h3>
<el-form label-width="80px" class="px-[10px]" @submit.prevent>
<el-form-item :label="t('submitBtnName')">
<el-input v-model.trim="diyStore.editComponent.submitBtn.text" :placeholder="t('btnNamePlaceholder')" clearable maxlength="10" show-word-limit />
</el-form-item>
<el-form-item :label="t('textColor')">
<el-color-picker v-model="diyStore.editComponent.submitBtn.color" />
</el-form-item>
<el-form-item :label="t('subTextBgColor')">
<el-color-picker v-model="diyStore.editComponent.submitBtn.bgColor" />
</el-form-item>
</el-form>
</div>
<div class="edit-attr-item-wrap">
<h3 class="mb-[10px]">{{ t('resetBtn') }}</h3>
<el-form label-width="80px" class="px-[10px]" @submit.prevent>
<el-form-item :label="t('carouselSearchTabControl')">
<el-switch v-model="diyStore.editComponent.resetBtn.control" />
</el-form-item>
<el-form-item :label="t('submitBtnName')">
<el-input v-model.trim="diyStore.editComponent.resetBtn.text" :placeholder="t('btnNamePlaceholder')" clearable maxlength="10" show-word-limit />
</el-form-item>
<el-form-item :label="t('textColor')">
<el-color-picker v-model="diyStore.editComponent.resetBtn.color" />
</el-form-item>
<el-form-item :label="t('subTextBgColor')">
<el-color-picker v-model="diyStore.editComponent.resetBtn.bgColor" />
</el-form-item>
</el-form>
</div>
</div>
<!-- 样式 -->
<div class="style-wrap" v-show="diyStore.editTab == 'style'">
<div class="edit-attr-item-wrap">
<h3 class="mb-[10px]">{{ t('btnStyle') }}</h3>
<el-form label-width="100px" class="px-[10px]">
<el-form-item :label="t('topRounded')">
<el-slider v-model="diyStore.editComponent.topElementRounded" show-input size="small" class="ml-[10px] diy-nav-slider" :max="50" />
</el-form-item>
<el-form-item :label="t('bottomRounded')">
<el-slider v-model="diyStore.editComponent.bottomElementRounded" show-input size="small" class="ml-[10px] diy-nav-slider" :max="50" />
</el-form-item>
</el-form>
</div>
<!-- 组件样式 -->
<slot name="style"></slot>
</div>
</template>
<script lang="ts" setup>
import { t } from '@/lang'
import { ref } from 'vue'
import useDiyStore from '@/stores/modules/diy'
const diyStore = useDiyStore()
diyStore.editComponent.ignore = ['componentBgUrl'] //
//
const btnPositionChangeFn = (e)=>{
if(e == 'hover_screen_bottom'){
diyStore.editComponent.margin.bottom = 0;
diyStore.editComponent.margin.both = 0;
diyStore.editComponent.margin.top = 0;
}
}
//
diyStore.editComponent.verify = (index: number) => {
const res = { code: true, message: '' }
if (diyStore.value[index].submitBtn.text == '') {
res.code = false;
res.message = t('submitBtnNamePlaceholder');
return res;
}
if (diyStore.value[index].resetBtn.text == '') {
res.code = false;
res.message = t('resetBtnNamePlaceholder');
return res;
}
return res
}
defineExpose({})
</script>
<style lang="scss" scoped></style>

View File

@ -0,0 +1,44 @@
<template>
<!-- 内容 -->
<div class="content-wrap" v-show="diyStore.editTab == 'content'">
<!-- 表单组件 字段内容设置 -->
<slot name="field"></slot>
todo 此处编写表格组件的属性
<!-- 表单组件 其他设置 -->
<slot name="other"></slot>
</div>
<!-- 样式 -->
<div class="style-wrap" v-show="diyStore.editTab == 'style'">
<!-- 表单组件 字段样式 -->
<slot name="style-field"></slot>
<!-- 组件样式 -->
<slot name="style"></slot>
</div>
</template>
<script lang="ts" setup>
import { t } from '@/lang'
import { ref } from 'vue'
import useDiyStore from '@/stores/modules/diy'
const diyStore = useDiyStore()
diyStore.editComponent.ignore = ['componentBgUrl'] //
//
diyStore.editComponent.verify = (index: number) => {
const res = { code: true, message: '' }
// todo
return res
}
defineExpose({})
</script>
<style lang="scss" scoped></style>

View File

@ -0,0 +1,84 @@
<template>
<!-- 内容 -->
<div class="content-wrap" v-show="diyStore.editTab == 'content'">
<!-- 表单组件 字段内容设置 -->
<slot name="field"></slot>
<el-form label-width="100px" class="px-[10px]" @submit.prevent>
<el-form-item :label="t('formPlaceholder')">
<el-input v-model.trim="diyStore.editComponent.placeholder" :placeholder="t('formPlaceholderTips')" clearable maxlength="15" show-word-limit />
</el-form-item>
<el-form-item>
<template #label>
<div class="flex items-center">
<span class="mr-[3px]">{{ t('defaultValue') }}</span>
<el-tooltip effect="light" :content="t('defaultValueTips')" placement="top">
<el-icon>
<QuestionFilled color="#999999" />
</el-icon>
</el-tooltip>
</div>
</template>
<el-input v-model.trim="diyStore.editComponent.field.default" :placeholder="t('defaultValuePlaceholder')" clearable maxlength="18" show-word-limit />
</el-form-item>
<el-form-item :label="t('rowCount')">
<el-input v-model.trim="diyStore.editComponent.rowCount" :placeholder="t('rowCountPlaceholder')" clearable maxlength="2" show-word-limit />
</el-form-item>
</el-form>
<!-- 表单组件 其他设置 -->
<slot name="other"></slot>
<!--<el-form label-width="100px" class="px-[10px]">
<el-form-item class="display-block">
<template #label>
<div class="flex items-center">
<span class="mr-[3px]">{{ t('隐私保护') }}</span>
<el-tooltip effect="light" placement="top">
<template #content>
<p>会自动将提交的个人信息做加密展示</p>
<p>适用于公开展示收集的数据且不暴露用户隐私</p>
</template>
<el-icon>
<QuestionFilled color="#999999" />
</el-icon>
</el-tooltip>
</div>
</template>
<el-switch v-model="diyStore.editComponent.field.privacyProtection" />
<div class="text-sm text-gray-400">{{ t('提交后自动隐藏文本,仅管理员可查看') }}</div>
</el-form-item>
</el-form>-->
</div>
<!-- 样式 -->
<div class="style-wrap" v-show="diyStore.editTab == 'style'">
<!-- 表单组件 字段样式 -->
<slot name="style-field"></slot>
<!-- 组件样式 -->
<slot name="style"></slot>
</div>
</template>
<script lang="ts" setup>
import { t } from '@/lang'
import { ref } from 'vue'
import useDiyStore from '@/stores/modules/diy'
const diyStore = useDiyStore()
diyStore.editComponent.ignore = ['componentBgUrl'] //
//
diyStore.editComponent.verify = (index: number) => {
const res = { code: true, message: '' }
// todo
return res
}
defineExpose({})
</script>
<style lang="scss" scoped></style>

View File

@ -0,0 +1,201 @@
<template>
<!-- 内容 -->
<div class="content-wrap" v-show="diyStore.editTab == 'content'">
<!-- 表单组件 字段内容设置 -->
<slot name="field"></slot>
<div class="edit-attr-item-wrap">
<h3 class="mb-[10px]">{{ t('startTime') }}</h3>
<el-form label-width="100px" class="px-[10px]" @submit.prevent>
<el-form-item :label="t('formPlaceholder')">
<el-input v-model.trim="diyStore.editComponent.start.placeholder" :placeholder="t('formPlaceholderTips')" clearable maxlength="15" show-word-limit />
</el-form-item>
<el-form-item :label="t('defaultValue')">
<el-switch v-model="diyStore.editComponent.start.defaultControl"/>
</el-form-item>
<el-form-item v-if="diyStore.editComponent.start.defaultControl">
<el-radio-group v-model="diyStore.editComponent.start.timeWay">
<el-radio label="current">{{ t('currentTime') }}</el-radio>
<el-radio label="diy">{{ t('diyTime') }}</el-radio>
</el-radio-group>
</el-form-item>
<el-form-item v-if="diyStore.editComponent.start.defaultControl && diyStore.editComponent.start.timeWay == 'diy'">
<el-time-picker v-model="diyStore.editComponent.field.default.start.date" :placeholder="t('startTimePlaceholder')" format="HH:mm" value-format="HH:mm" @change="startTimePickerChange" />
</el-form-item>
</el-form>
</div>
<div class="edit-attr-item-wrap">
<h3 class="mb-[10px]">{{ t('endTime') }}</h3>
<el-form label-width="100px" class="px-[10px]" @submit.prevent>
<el-form-item :label="t('formPlaceholder')">
<el-input v-model.trim="diyStore.editComponent.end.placeholder" :placeholder="t('formPlaceholderTips')" clearable maxlength="15" show-word-limit />
</el-form-item>
<el-form-item :label="t('defaultValue')">
<el-switch v-model="diyStore.editComponent.end.defaultControl" />
</el-form-item>
<el-form-item v-if="diyStore.editComponent.end.defaultControl">
<el-radio-group v-model="diyStore.editComponent.end.timeWay">
<el-radio label="current">{{ t('currentTime') }}</el-radio>
<el-radio label="diy">{{ t('diyTime') }}</el-radio>
</el-radio-group>
</el-form-item>
<el-form-item v-if="diyStore.editComponent.end.defaultControl && diyStore.editComponent.end.timeWay == 'diy'">
<el-time-picker :disabled-hours="disabledHours" :disabled-minutes="disabledMinutes" v-model="diyStore.editComponent.field.default.end.date" :placeholder="t('endTimePlaceholder')" format="HH:mm" value-format="HH:mm" />
</el-form-item>
</el-form>
</div>
<!-- 表单组件 其他设置 -->
<slot name="other"></slot>
</div>
<!-- 样式 -->
<div class="style-wrap" v-show="diyStore.editTab == 'style'">
<div class="edit-attr-item-wrap">
<h3 class="mb-[10px]">{{ t('textStyle') }}</h3>
<el-form label-width="80px" class="px-[10px]">
<el-form-item :label="t('textFontSize')">
<el-slider v-model="diyStore.editComponent.fontSize" show-input size="small" class="ml-[10px] diy-nav-slider" :min="12" :max="18" />
</el-form-item>
<el-form-item :label="t('textFontWeight')">
<el-radio-group v-model="diyStore.editComponent.fontWeight">
<el-radio :label="'normal'">{{ t('fontWeightNormal') }}</el-radio>
<el-radio :label="'bold'">{{ t('fontWeightBold') }}</el-radio>
</el-radio-group>
</el-form-item>
<el-form-item :label="t('textColor')">
<el-color-picker v-model="diyStore.editComponent.textColor" />
</el-form-item>
</el-form>
</div>
<!-- 组件样式 -->
<slot name="style"></slot>
</div>
</template>
<script lang="ts" setup>
import { t } from '@/lang'
import { ref,onMounted } from 'vue'
import useDiyStore from '@/stores/modules/diy'
const diyStore = useDiyStore()
diyStore.editComponent.ignore = ['componentBgUrl'] //
//
diyStore.editComponent.verify = (index: number) => {
const res = { code: true, message: '' }
let starTime = diyStore.value[index].field.default.start.date;
let endTime = diyStore.value[index].field.default.end.date;
const today = new Date();
const hours = String(today.getHours()).padStart(2, '0');
const minutes = String(today.getMinutes()).padStart(2, '0');
if(diyStore.editComponent.start.timeWay == 'current'){
starTime = `${hours}:${minutes}`;
}
if(diyStore.editComponent.end.timeWay == 'current'){
endTime = `${hours}:${minutes}`;
}
if(diyStore.editComponent.start.defaultControl && starTime == ''){
res.code = false
res.message = t('startTimeTips')
return res
}
if(diyStore.editComponent.end.defaultControl && endTime == ''){
res.code = false
res.message = t('endTimeTips')
return res
}
if(diyStore.editComponent.start.defaultControl && diyStore.editComponent.end.defaultControl && timeInvertSecond(starTime) > timeInvertSecond(endTime)){
res.code = false
res.message = t('startEndTimeTips')
return res
}
return res
}
onMounted(() => {
const today = new Date();
const hours = String(today.getHours()).padStart(2, '0');
const minutes = String(today.getMinutes()).padStart(2, '0');
if(!diyStore.editComponent.field.default.start.date){
diyStore.editComponent.field.default.start.date = `${hours}:${minutes}`;
diyStore.editComponent.field.default.start.timestamp = timeInvertSecond(`${hours}:${minutes}`);
}
if(!diyStore.editComponent.field.default.end.date){
let endDate = new Date();
endDate.setHours(today.getHours(), today.getMinutes() + 10, 0, 0); // 10
const endHours = String(endDate.getHours()).padStart(2, '0');
const endMinutes = String(endDate.getMinutes()).padStart(2, '0');
diyStore.editComponent.field.default.end.date = `${endHours}:${endMinutes}`;
diyStore.editComponent.field.default.end.timestamp = timeInvertSecond(`${endHours}:${endMinutes}`);
}
});
//
const startTimePickerChange = (e) => {
diyStore.editComponent.field.default.start.timestamp = timeInvertSecond(e);
const startTimeArr = e.split(":");
const date = new Date();
date.setHours(parseInt(startTimeArr[0]), parseInt(startTimeArr[1]), 0, 0);
date.setMinutes(date.getMinutes() + 10);
const updatedEndTime = `${String(date.getHours()).padStart(2, '0')}:${String(date.getMinutes()).padStart(2, '0')}`;
diyStore.editComponent.field.default.end.date = updatedEndTime;
diyStore.editComponent.field.default.end.timestamp = timeInvertSecond(updatedEndTime);
}
//
const endTimePickerChange = (e)=>{
diyStore.editComponent.field.default.end.timestamp = timeInvertSecond(e);
}
const disabledHours = () => {
let timeArr = diyStore.editComponent.field.default.start.date.split(":")
return makeRange(0,timeArr[0]);
}
const disabledMinutes = (hour: number) => {
let timeArr = diyStore.editComponent.field.default.start.date.split(":")
return makeRange(0,timeArr[1]);
}
const makeRange = (start: number, end: number) => {
const result: number[] = []
for (let i = start; i < end; i++) {
result.push(i)
}
return result
}
const timeInvertSecond = (time:any)=>{
let arr = time.split(":");
let num = 0;
if(arr[0]){
num += arr[0] * 60 * 60;
}
if(arr[1]){
num += arr[1] * 60;
}
if(arr[2]){
num += arr[2];
}
return num;
}
defineExpose({})
</script>
<style lang="scss" scoped></style>

View File

@ -0,0 +1,89 @@
<template>
<!-- 内容 -->
<div class="content-wrap" v-show="diyStore.editTab == 'content'">
<!-- 表单组件 字段内容设置 -->
<slot name="field"></slot>
<el-form label-width="100px" class="px-[10px]" @submit.prevent>
<el-form-item :label="t('formPlaceholder')">
<el-input v-model.trim="diyStore.editComponent.placeholder" :placeholder="t('formPlaceholderTips')" clearable maxlength="15" show-word-limit />
</el-form-item>
<el-form-item :label="t('defaultValue')">
<el-switch v-model="diyStore.editComponent.defaultControl" @change="changeDateDefaultControl" />
</el-form-item>
<el-form-item v-if="diyStore.editComponent.defaultControl">
<el-radio-group v-model="diyStore.editComponent.timeWay">
<el-radio label="current">{{ t('currentTime') }}</el-radio>
<el-radio label="diy">{{ t('diyTime') }}</el-radio>
</el-radio-group>
</el-form-item>
<el-form-item v-if="diyStore.editComponent.defaultControl && diyStore.editComponent.timeWay == 'diy'">
<el-time-picker v-model="diyStore.editComponent.field.default" :placeholder="t('timePlaceholder')" format="HH:mm" value-format="HH:mm" />
</el-form-item>
</el-form>
<!-- 表单组件 其他设置 -->
<slot name="other"></slot>
</div>
<!-- 样式 -->
<div class="style-wrap" v-show="diyStore.editTab == 'style'">
<!-- 表单组件 字段样式 -->
<slot name="style-field"></slot>
<!-- 组件样式 -->
<slot name="style"></slot>
</div>
</template>
<script lang="ts" setup>
import { t } from '@/lang'
import { ref,watch,onMounted } from 'vue'
import useDiyStore from '@/stores/modules/diy'
const diyStore = useDiyStore()
diyStore.editComponent.ignore = ['componentBgUrl'] //
//
diyStore.editComponent.verify = (index: number) => {
const res = { code: true, message: '' }
return res
}
onMounted(() => {
//
if (!diyStore.editComponent.field.default) {
const today = new Date();
const hours = String(today.getHours()).padStart(2, '0');
const minutes = String(today.getMinutes()).padStart(2, '0');
diyStore.editComponent.field.default = `${hours}:${minutes}`;
}
});
const changeDateDefaultControl = (val: any) => {
if (val) {
const today = new Date();
const hours = String(today.getHours()).padStart(2, '0');
const minutes = String(today.getMinutes()).padStart(2, '0');
diyStore.editComponent.field.default = `${hours}:${minutes}`;
}
}
watch(
() => diyStore.editComponent.timeWay,
(newVal) => {
const today = new Date();
const hours = String(today.getHours()).padStart(2, '0');
const minutes = String(today.getMinutes()).padStart(2, '0');
diyStore.editComponent.field.default = `${hours}:${minutes}`;
}
);
defineExpose({})
</script>
<style lang="scss" scoped></style>

View File

@ -0,0 +1,42 @@
<template>
<!-- 内容 -->
<div class="content-wrap" v-show="diyStore.editTab == 'content'">
<!-- 表单组件 字段内容设置 -->
<slot name="field"></slot>
<el-form label-width="100px" class="px-[10px]">
<el-form-item :label="t('上传方式')">
<el-radio-group v-model="diyStore.editComponent.uploadMode">
<el-radio label="shoot_and_album">{{ t('拍摄和相册') }}</el-radio>
<el-radio label="shoot_only">{{ t('只允许拍摄') }}</el-radio>
</el-radio-group>
</el-form-item>
</el-form>
</div>
<!-- 样式 -->
<div class="style-wrap" v-show="diyStore.editTab == 'style'">
<!-- 表单组件 字段样式 -->
<slot name="style-field"></slot>
<!-- 组件样式 -->
<slot name="style"></slot>
</div>
</template>
<script lang="ts" setup>
import { t } from '@/lang'
import { ref } from 'vue'
import useDiyStore from '@/stores/modules/diy'
const diyStore = useDiyStore()
diyStore.editComponent.ignore = ['componentBgUrl'] //
defineExpose({})
</script>
<style lang="scss" scoped></style>

View File

@ -0,0 +1,48 @@
<template>
<!-- 内容 -->
<div class="content-wrap" v-show="diyStore.editTab == 'content'">
<!-- 表单组件 字段内容设置 -->
<slot name="field"></slot>
<el-form label-width="100px" class="px-[10px]" @submit.prevent>
<el-form-item :label="t('formPlaceholder')">
<el-input v-model.trim="diyStore.editComponent.placeholder" :placeholder="t('formPlaceholderTips')" clearable maxlength="15" show-word-limit />
</el-form-item>
</el-form>
<!-- 表单组件 其他设置 -->
<slot name="other"></slot>
</div>
<!-- 样式 -->
<div class="style-wrap" v-show="diyStore.editTab == 'style'">
<!-- 表单组件 字段样式 -->
<slot name="style-field"></slot>
<!-- 组件样式 -->
<slot name="style"></slot>
</div>
</template>
<script lang="ts" setup>
import { t } from '@/lang'
import { ref } from 'vue'
import useDiyStore from '@/stores/modules/diy'
const diyStore = useDiyStore()
diyStore.editComponent.ignore = ['componentBgUrl'] //
//
diyStore.editComponent.verify = (index: number) => {
const res = { code: true, message: '' }
// todo
return res
}
defineExpose({})
</script>
<style lang="scss" scoped></style>

View File

@ -0,0 +1,144 @@
<template>
<div>
<el-dialog v-model="showDialog" :title="t('formPromotion')" width="500px" :destroy-on-close="true">
<el-tabs v-model="channel" class="mb-[10px]">
<el-tab-pane label="H5" name="h5"></el-tab-pane>
<el-tab-pane label="微信小程序" name="weapp"></el-tab-pane>
</el-tabs>
<div class="promote-flex flex" v-if="channel == 'h5'">
<div class="promote-img flex justify-center items-center bg-[#f8f8f8] w-[150px] h-[150px]">
<el-image :src="wapImage" />
</div>
<div class="px-[20px] flex-1">
<div class="mb-[10px]">{{ t('promoteUrl') }}</div>
<el-input class="mb-[10px]" readonly :value="wapPreview">
<template #append>
<el-button @click="copyEvent(wapPreview)">{{ t('copy') }}</el-button>
</template>
</el-input>
<a class="text-primary" :href="wapImage" download>{{ t('downLoadQRCode') }}</a>
</div>
</div>
<div class="promote-flex flex" v-if="channel == 'weapp'">
<div class="promote-img flex justify-center items-center bg-[#f8f8f8] w-[150px] h-[150px]">
<el-image :src="img(weappData.path)" v-if="weappData.path" class="w-[150px] h-[150px]">
<template #error>
<div class="w-[150px] h-[150px] text-[14px] text-center leading-[150px]">{{ t('configureFailed') }}</div>
</template>
</el-image>
<div v-else class="w-[150px] h-[150px] text-[14px] text-center leading-[150px]">{{ t('configureFailed') }}</div>
</div>
<div class="px-[20px] flex-1">
<a class="text-primary" :href="img(weappData.path)" target="_blank" download>{{ t('downLoadQRCode') }}</a>
</div>
</div>
</el-dialog>
</div>
</template>
<script lang="ts" setup>
import { t } from '@/lang'
import { ref, reactive, watch } from 'vue'
import { ElMessage } from 'element-plus'
import QRCode from 'qrcode'
import storage from '@/utils/storage'
import { useClipboard } from '@vueuse/core'
import { getUrl } from '@/app/api/sys'
import { getDiyFormQrcode } from '@/app/api/diy_form'
import { img } from '@/utils/common'
const form: any = reactive({})
const showDialog = ref(false)
const channel = ref('h5')
const wapUrl = ref('')
const wapDomain = ref('')
const wapImage = ref('')
const wapPreview = ref('')
const page = ref('')
const weappData = reactive({
path: ''
})
getUrl().then((res: any) => {
wapUrl.value = res.data.wap_url
//
if (import.meta.env.MODE == 'production') return
wapDomain.value = res.data.wap_domain
// envwap
if (wapDomain.value) {
wapUrl.value = wapDomain.value + '/wap'
}
const wapDomainStorage = storage.get('wap_domain')
if (wapDomainStorage) {
wapUrl.value = wapDomainStorage
}
})
const loadQrcode = () => {
wapPreview.value = `${wapUrl.value}${page.value}`
// errorCorrectionLevelLH()
QRCode.toDataURL(wapPreview.value, { errorCorrectionLevel: 'L', margin: 0, width: 120 }).then(url => {
wapImage.value = url
})
}
const show = (data: any) => {
channel.value = 'h5';
Object.assign(form, data)
page.value = `/app/pages/index/diy_form?form_id=${form.form_id}`
loadQrcode()
getDiyFormQrcodeFn();
showDialog.value = true
}
const getDiyFormQrcodeFn = ()=>{
getDiyFormQrcode({
form_id:form.form_id
}).then((res:any)=>{
if(res.data) {
weappData.path = res.data.path;
}
})
}
//
const { copy, isSupported, copied } = useClipboard()
const copyEvent = (text: string) => {
if (!isSupported.value) {
ElMessage({
message: t('notSupportCopy'),
type: 'warning'
})
}
copy(text)
}
watch(copied, () => {
if (copied.value) {
ElMessage({
message: t('copySuccess'),
type: 'success'
})
}
})
defineExpose({
showDialog,
show
})
</script>
<style lang="scss" scoped></style>

View File

@ -0,0 +1,435 @@
<template>
<div>
<el-dialog v-model="showDialog" :title="t('submirSuccess')" width="850px" :close-on-press-escape="true" :destroy-on-close="true" :close-on-click-modal="false">
<div class="flex flex-1 mt-[24px] mx-[24px] mb-0">
<div class="preview-wrap">
<div class="absolute z-1 left-0 top-0">
<img src="@/app/assets/images/diy_form/mobile_tabbar.png" class="w-[324px]" />
</div>
<div class="absolute z-1 left-0 bottom-0">
<img src="@/app/assets/images/diy_form/mobile_bottom.png" class="w-[324px]" />
</div>
<div class="page-wrap">
<div class="px-[13px] flex flex-col items-center flex-1">
<div class="flex items-center justify-center w-[48px] h-[48px] text-[40px] p-[4px] mt-[32px] mb-[16px] mx-auto rounded-[50%]">
<el-icon><SuccessFilled color="#20bf64" /></el-icon>
</div>
<div class="record-name">
<span class="text-[#1E1E1E] font-bold text-[24px]" v-if="formData.tips_type == 'default'">{{ t('writeSuccess') }}</span>
<span class="text-[#1E1E1E] font-bold text-[16px]" v-else-if="formData.tips_type == 'diy'">{{ formData.tips_text ? formData.tips_text : '填写成功' }}</span>
</div>
<div class="to-detail">
<div class="text-[14px] mt-[16px] py-[4px] px[8px] text-[#576b95]">{{ t('viewFillingDetails') }}</div>
</div>
</div>
<div class="relative pt-[8px] pb-[48px] h-[112px]">
<div v-if="formData.success_after_action.finish" class="!mt-[16px] rounded-[3px] mx-auto text-[15px] w-[100px] min-w-[160px] h-[32px] leading-[32px] text-center max-w-[274px] truncate bg-[#20bf64] text-[#ffffff]">
<div class="text-[15px]">{{ t('finish') }}</div>
</div>
<div v-if="formData.success_after_action.goback" class="!mt-[16px] rounded-[3px] mx-auto text-[15px] w-[100px] min-w-[160px] h-[32px] leading-[32px] text-center max-w-[274px] truncate bg-[#f2f2f2] text-[#353535]">
<div class="text-[14px]">{{ t('back') }}</div>
</div>
</div>
</div>
<!-- 核销凭证 todo 后续完善 -->
<!-- <div class="page-wrap verify-voucher-wrap" style="display:none;">-->
<!-- <div class="tips-wrap">感谢你的填写以下是你的核销凭证</div>-->
<!-- <div class="qrcode-wrap">-->
<!-- <div class="text-[14px] text-[#333]">请妥善保存你的核销凭证</div>-->
<!-- <div class="text-[20px] font-bold text-[#333] my-[10px]">现场出示凭证</div>-->
<!-- <el-image class="w-[180px]" :src="wapImage" />-->
<!-- <div class="text-primary mt-[10px]">保存凭证</div>-->
<!-- </div>-->
<!-- <div class="relative pt-[8px] pb-[48px] h-[112px]">-->
<!-- <div class="!mt-[16px] rounded-[3px] mx-auto text-[15px] w-[100px] min-w-[160px] h-[32px] leading-[32px] text-center max-w-[274px] truncate bg-[#20bf64] text-[#ffffff]">-->
<!-- <div class="text-[15px]">返回二维码</div>-->
<!-- </div>-->
<!-- <div class="!mt-[16px] rounded-[3px] mx-auto text-[15px] w-[100px] min-w-[160px] h-[32px] leading-[32px] text-center max-w-[274px] truncate bg-[#fff] text-[#353535]">-->
<!-- <div class="text-[14px]">完成</div>-->
<!-- </div>-->
<!-- </div>-->
<!-- <div class="text-[14px] mt-[16px] py-[4px] px[8px] text-[#576b95]">查看填写详情</div>-->
<!-- </div>-->
</div>
<div class="flex-1">
<div class="item-wrap">
<div class="text-[16px] h-[24px] font-bold text-[#262626] mb-[16px] w-[140px] mr-[32px] flex-shrink-0">{{ t('afterSubmission') }}</div>
<el-radio-group v-model="formData.submit_after_action" class="!block">
<el-radio label="text" class="!flex">
<span class="mr-[3px]">{{ t('displayTextMessages') }}</span>
<el-tooltip effect="light" placement="top">
<template #content>
<p>{{ t('displayTextMessagesTips') }}</p>
</template>
<el-icon>
<QuestionFilled color="#999999" />
</el-icon>
</el-tooltip>
</el-radio>
<!-- todo 后续完善 -->
<!-- <el-radio label="voucher" class="!flex">-->
<!-- <span class="mr-[3px]">{{ t('获取核销凭证') }}</span>-->
<!-- <el-tooltip effect="light" placement="top">-->
<!-- <template #content>-->
<!-- <p>{{ t('提交后页面会将提交的表单记录内容生成二维码并展示,可选择设置两种不同的二维码内容。适合核销、数据录入等场景。') }}</p>-->
<!-- </template>-->
<!-- <el-icon>-->
<!-- <QuestionFilled color="#999999" />-->
<!-- </el-icon>-->
<!-- </el-tooltip>-->
<!-- </el-radio>-->
</el-radio-group>
</div>
<div class="item-wrap" v-if="formData.submit_after_action == 'text'">
<div class="text-[16px] h-[24px] font-bold text-[#262626] mb-[16px] w-[140px] mr-[32px] flex-shrink-0">{{ t('promptText') }}</div>
<div>
<el-radio-group v-model="formData.tips_type" class="!block">
<el-radio label="default" class="!block">
<span class="mr-[3px]">{{ t('defaultPrompt') }}</span>
<span class="!text-[#999] text-[12px] ml-[8px]">{{ t('defaultPromptTips') }}</span>
</el-radio>
<el-radio label="diy" class="!block">{{ t('diyPrompt') }}</el-radio>
</el-radio-group>
<el-input v-if="formData.tips_type == 'diy'" v-model.trim="formData.tips_text" :placeholder="t('tipsTextPlaceholder')" class="w-[350px]" maxlength="30" clearable show-word-limit />
</div>
</div>
<!-- 核销凭证 todo 后续完善 -->
<template v-else-if="formData.submit_after_action == 'voucher'">
<div class="item-wrap">
<div class="text-[16px] h-[24px] font-bold text-[#262626] mb-[16px] w-[140px] mr-[32px] flex-shrink-0">{{ t('validityPeriodOfVoucher') }}</div>
<div>
<el-radio-group v-model="formData.time_limit_type" class="!block">
<el-radio label="no_limit" class="!block">{{ t('noLimit') }}</el-radio>
<el-radio label="specify_time" class="!block">
<span class="mr-[3px]">{{ t('specifyTime') }}</span>
<el-tooltip effect="light" placement="top">
<template #content>
<p>{{ t('specifyTimeTips') }}</p>
</template>
<el-icon>
<QuestionFilled color="#999999" />
</el-icon>
</el-tooltip>
</el-radio>
<el-radio label="submission_time" class="!block">
<span class="mr-[3px]">{{ t('submissionTime') }}</span>
<el-tooltip effect="light" placement="top">
<template #content>
<p class="w-[250px]">{{ t('submissionTimeTips') }}</p>
</template>
<el-icon>
<QuestionFilled color="#999999" />
</el-icon>
</el-tooltip>
</el-radio>
</el-radio-group>
<el-date-picker v-if="formData.time_limit_type == 'specify_time'" v-model="formData.validity_time" type="datetimerange" range-separator="至" start-placeholder="开始时间" end-placeholder="结束时间" />
<div class="flex items-center mt-[5px]" v-if="formData.time_limit_type == 'submission_time'">
<span>{{ t('afterSubmissionRecords') }}</span>
<!-- <div class="flex items-center px-[5px]">-->
<!-- v-model.trim="formData.length"-->
<el-input v-model.trim="formData.submission_time_value" @keyup="filterNumber($event)" size="small" clearable class="!w-[100px] px-[5px]" maxlength="3" />
<el-select v-model="formData.timeUnit" clearable class="!w-[100px] pr-[5px]" size="small">
<el-option v-for="(item, index) in validityOptions" :key="item.value" :label="item.text" :value="item.value"></el-option>
</el-select>
<span>{{ t('effective') }}</span>
</div>
</div>
</div>
<div class="item-wrap">
<div class="text-[16px] h-[24px] font-bold text-[#262626] mb-[16px] w-[140px] mr-[32px] flex-shrink-0">{{ t('voucherStyle') }}</div>
<div>
<el-form-item :label="t('titleAboveTheCode')">
<!-- v-model.trim="formData.active_name"-->
<el-input clearable :placeholder="t('titleAboveTheCodePlaceholder')" class="input-width" :maxlength="20" />
</el-form-item>
<el-form-item :label="t('contentAboveTheCode')">
<el-input clearable :placeholder="t('contentAboveTheCodePlaceholder')" class="input-width" :maxlength="20" />
<div>
<span class="text-primary cursor-pointer mr-[10px]">{{ t('addLinefeeds') }}</span>
<span class="text-primary cursor-pointer">{{ t('addFields') }}</span>
</div>
</el-form-item>
<el-form-item :label="t('contentBelowTheCode')">
<div class="block">
<el-checkbox class="!block" :label="t('submissionRecordTime')" value="" />
<el-checkbox class="!block !h-[20px]" :label="t('currentTime')" value="" />
<div class="text-[#999] ml-[22px]">{{ t('currentTimeTips') }}</div>
<el-checkbox class="!block" :label="t('dispalyPromptText')" value="" />
<el-input class="ml-[22px]" :rows="4" type="textarea" :placeholder="t('tipsTextPlaceholder')" maxlength="100" />
<el-checkbox class="!block" :label="t('voucherDeadline')" value="" />
<el-checkbox class="!block" :label="t('saveVoucher')" value="" />
</div>
</el-form-item>
</div>
</div>
</template>
<!-- todo 后续完善 -->
<div class="item-wrap">
<div class="text-[16px] h-[24px] font-bold text-[#262626] mb-[16px] w-[140px] mr-[32px] flex-shrink-0">
<span>{{ t('subsequentPperationButtons') }}</span>
<!-- <p class="text-[12px] text-[#999] mt-[4px] font-normal">最多选择2个</p>-->
</div>
<div class="content-list-wrap">
<!-- <el-checkbox-group :min="1" :max="2">-->
<!-- <el-checkbox v-model="formData.success_after_action.share" label="转发填写内容" value="share">-->
<!-- <div class="text-[#333]">转发填写内容</div>-->
<!-- </el-checkbox>-->
<!-- <p class="text-[#999] text-[12px] pl-[24px] mt-[4px]">提交表单后可转发给微信好友查看支持按钮文案自定义提醒填表人转发给特定人员查看</p>-->
<el-checkbox v-model="formData.success_after_action.finish" :label="t('finish')" value="finish">
<div class="text-[#333]">{{ t('finish') }}</div>
</el-checkbox>
<p class="text-[#999] text-[12px] pl-[24px] mt-[4px]">{{ t('finishTips') }}</p>
<el-checkbox v-model="formData.success_after_action.goback" :label="t('back')" value="goback">
<div class="text-[#333]">{{ t('back') }}</div>
</el-checkbox>
<p class="text-[#999] text-[12px] pl-[24px] mt-[4px]">{{ t('backTips') }}</p>
<!-- </el-checkbox-group>-->
</div>
</div>
</div>
</div>
<template #footer>
<div class="dialog-footer">
<el-button @click="showDialog = false">{{ t('cancel') }}</el-button>
<el-button type="primary" @click="confirm">{{ t('save') }}</el-button>
</div>
</template>
</el-dialog>
</div>
</template>
<script lang="ts" setup>
import { t } from '@/lang'
import { ref, reactive } from 'vue'
import { ElMessage } from 'element-plus'
import QRCode from 'qrcode'
import storage from '@/utils/storage'
import { filterNumber } from '@/utils/common'
import { getUrl } from '@/app/api/sys'
import { getFormSubmitConfig,editDiyFormSubmitConfig } from '@/app/api/diy_form'
const showDialog = ref(false)
const repeat = ref(false)
/**
* 表单数据
*/
const initialFormData = {
id: 0,
form_id: 0,
submit_after_action: 'text', // textvoucher
tips_type: 'default', // defaultdiy
tips_text: '', //
time_limit_type: 'no_limit', // no_limitspecify_timesubmission_time
// json todo
time_limit_rule: {
validity_time: [], //
submission_time_value: '', //
timeUnit: 'day', //
},
// json todo
voucher_content_rule: {},
//
success_after_action: {
share: false, //
finish: true, //
goback: true, //
}
}
const formData: Record<string, any> = reactive({ ...initialFormData })
const wapUrl = ref('')
const wapDomain = ref('')
const wapImage = ref('')
const wapPreview = ref('')
const page = ref('')
//
const validityOptions = reactive([
{
text:'天',
value:'day'
},
{
text:'周',
value:'week'
},
{
text:'月',
value:'month'
},
{
text:'年',
value:'year'
},
{
text:'分钟',
value:'minutes'
}
])
// getUrl().then((res: any) => {
// wapUrl.value = res.data.wap_url
//
// //
// if (import.meta.env.MODE == 'production') return
//
// wapDomain.value = res.data.wap_domain
//
// // envwap
// if (wapDomain.value) {
// wapUrl.value = wapDomain.value + '/wap'
// }
//
// const wapDomainStorage = storage.get('wap_domain')
// if (wapDomainStorage) {
// wapUrl.value = wapDomainStorage
// }
// })
const loadQrcode = () => {
wapPreview.value = `${wapUrl.value}${page.value}`
// errorCorrectionLevelLH()
QRCode.toDataURL(wapPreview.value, { errorCorrectionLevel: 'L', margin: 0, width: 120 }).then(url => {
wapImage.value = url
})
}
const emit = defineEmits(['complete'])
const setFormData = async (row: any = null) => {
Object.assign(formData, initialFormData)
if (row) {
const data = await (await getFormSubmitConfig(row.form_id)).data
if (data) {
Object.keys(formData).forEach((key: string) => {
if (data[key] != undefined) formData[key] = data[key]
})
}
// todo
// page.value = `/app/pages/index/diy_form?form_id=${formData.form_id}`
// loadQrcode()
}
}
/**
* 确认
*/
const confirm = () => {
if(formData.tips_type == 'diy' && !formData.tips_text){
ElMessage.error('提示不能为空')
return
}
if (repeat.value) return;
repeat.value = true
const data = formData
editDiyFormSubmitConfig(data).then(res => {
repeat.value = false
showDialog.value = false
emit('complete')
}).catch(err => {
repeat.value = false
})
}
defineExpose({
showDialog,
setFormData
})
</script>
<style lang="scss" scoped>
.preview-wrap {
position: relative;
width: 324px;
min-height: 555px;
padding: 82px 12px 20px;
background-size: 100%;
background-repeat: repeat-y;
background-image: url(../../../../app/assets/images/diy_form/mobile_line.png);
border-radius: 38px;
overflow: hidden;
box-shadow: none;
background-color: #fff !important;
margin-right: 24px;
overflow-y: auto;
.page-wrap {
position: relative;
display: -ms-flexbox;
display: flex;
-ms-flex-direction: column;
flex-direction: column;
overflow-y: auto;
overflow-x: hidden;
text-align: center;
min-height: 548px;
height: auto;
}
.verify-voucher-wrap {
background-color: #f4f4f4;
.tips-wrap{
font-size: 15px;
font-weight: 400;
line-height: 21px;
color: rgba(0,0,0,.65);
margin: 20px 10px;
}
.qrcode-wrap{
border-radius: 12px;
margin: 0 20px;
background: #fff;
padding: 20px 10px 10px;
}
}
}
.item-wrap {
padding: 20px 24px 24px;
background-color: #fff;
border-radius: 2px;
display: flex;
position: relative;
&:after {
content: "";
display: block;
height: 1px;
width: calc(100% - 48px);
background-color: hsla(210, 8%, 51%, .13);
position: absolute;
left: 24px;
bottom: 0;
}
&:last-child:after {
display: none;
}
}
</style>

View File

@ -0,0 +1,341 @@
<template>
<el-dialog v-model="showDialog" :title="t('writeSet')" width="600px" class="diy-dialog-wrap" :close-on-press-escape="true" :destroy-on-close="true" :close-on-click-modal="false">
<el-form :model="formData" label-width="120px" ref="formRef" :rules="formRules" class="page-form" v-loading="loading">
<!-- <el-form-item :label="t('填写方式')">-->
<!-- <el-radio-group v-model="formData.write_way">-->
<!-- <el-radio label="no_limit">{{t('不限制')}}</el-radio>-->
<!-- <el-radio label="scan">{{t('仅限扫一扫')}}</el-radio>-->
<!-- <el-radio label="url">{{t('仅限链接进入')}}</el-radio>-->
<!-- </el-radio-group>-->
<!-- </el-form-item>-->
<el-form-item :label="t('joinMemberType')">
<el-radio-group v-model="formData.join_member_type">
<el-radio label="all_member">{{t('allMember')}}</el-radio>
<el-radio label="selected_member_level">{{t('selectedMemberLevel')}}</el-radio>
<el-radio label="selected_member_label">{{t('selectedMemberLabel')}}</el-radio>
</el-radio-group>
</el-form-item>
<!-- 会员标签 -->
<el-form-item :label="t('memberLabel')" prop="label_ids" v-if="formData.join_member_type=='selected_member_label'">
<el-select v-model="formData.label_ids" clearable multiple :placeholder="t('memberLabelPlaceholder')" class="input-width">
<el-option :label="item['label_name']" :value="item['label_id']" v-for="(item, index) in labelSelectData" :key="index" />
</el-select>
</el-form-item>
<!-- 会员等级 -->
<el-form-item :label="t('memberLevel')" prop="level_ids" v-if="formData.join_member_type=='selected_member_level'">
<el-select v-model="formData.level_ids" clearable multiple :placeholder="t('memberLevelPlaceholder')" class="input-width">
<el-option :label="item['level_name']" :value="item['level_id']" v-for="(item, index) in levelSelectData" :key="index" />
</el-select>
</el-form-item>
<el-form-item :label="t('apieceFillQuantity')" :class="{ '!mb-[5px]' : formData.member_write_type == 'diy' }">
<el-radio-group v-model="formData.member_write_type">
<el-radio label="no_limit">{{t('noLimit')}}</el-radio>
<el-radio label="diy">{{t('diy')}}</el-radio>
</el-radio-group>
</el-form-item>
<el-form-item label=" " v-if="formData.member_write_type == 'diy'" prop="member_write_rule">
<div class="flex items-center">
<span></span>
<el-input v-model.trim="formData.member_write_rule.time_value" @keyup="filterNumber($event)" size="small" class="!w-[50px] px-[5px]" maxlength="3" />
<el-select v-model="formData.member_write_rule.time_unit" class="!w-[60px] pr-[5px]" size="small">
<el-option v-for="(item, index) in validityOptions" :key="item.value" :label="item.text" :value="item.value"></el-option>
</el-select>
<span>可填写</span>
<el-input v-model.trim="formData.member_write_rule.num" @keyup="filterNumber($event)" size="small" class="!w-[50px] px-[5px]" maxlength="3" />
<span></span>
</div>
</el-form-item>
<el-form-item :label="t('fillQuantityTotal')" :class="{ '!mb-[5px]' : formData.form_write_type == 'diy' }">
<el-radio-group v-model="formData.form_write_type">
<el-radio label="no_limit">{{t('noLimit')}}</el-radio>
<el-radio label="diy">{{t('diy')}}</el-radio>
</el-radio-group>
</el-form-item>
<el-form-item label=" " v-if="formData.form_write_type == 'diy'" prop="form_write_rule">
<div class="flex items-center">
<span></span>
<el-input v-model.trim="formData.form_write_rule.time_value" @keyup="filterNumber($event)" size="small" class="!w-[50px] px-[5px]" maxlength="3" />
<el-select v-model="formData.form_write_rule.time_unit" class="!w-[60px] pr-[5px]" size="small">
<el-option v-for="(item, index) in validityOptions" :key="item.value" :label="item.text" :value="item.value"></el-option>
</el-select>
<span>可填写</span>
<el-input v-model.trim="formData.form_write_rule.num" @keyup="filterNumber($event)" size="small" class="!w-[50px] px-[5px]" maxlength="3" />
<span class="mr-[5px]"></span>
<el-tooltip effect="light" placement="top">
<template #content>
<p class="w-[250px]">{{ t('writeTips') }}</p>
</template>
<el-icon>
<QuestionFilled color="#999999" />
</el-icon>
</el-tooltip>
</div>
</el-form-item>
<el-form-item :label="t('fillInTheTimePeriod')" prop="time_limit_rule">
<el-radio-group v-model="formData.time_limit_type">
<el-radio label="no_limit">{{t('noLimit')}}</el-radio>
<el-radio label="specify_time">{{t('setSpecifyTime')}}</el-radio>
<el-radio label="open_day_time">{{t('openDayTime')}}</el-radio>
</el-radio-group>
<el-date-picker v-if="formData.time_limit_type == 'specify_time'" v-model="formData.time_limit_rule.specify_time" type="datetimerange" range-separator="至" start-placeholder="开始时间" end-placeholder="结束时间" />
<div class="flex items-center mt-[5px]" v-if="formData.time_limit_type == 'open_day_time'">
<span class="mr-[5px]">每天</span>
<el-time-picker class="!w-[180px]" v-model="formData.time_limit_rule.open_day_time" format="HH:mm" value-format="HH:mm" is-range range-separator="-" start-placeholder="开始时间" end-placeholder="结束时间" />
<span class="ml-[5px]">可填写</span>
</div>
</el-form-item>
<!-- <el-form-item :label="t('允许修改内容')" class="display-block">-->
<!-- <el-switch v-model="formData.is_allow_update_content" :active-value="1" :inactive-value="0" />-->
<!-- <div class="text-sm text-gray-400">{{ t('开启后,填表人可以修改自己填写的内容。') }}</div>-->
<!-- </el-form-item>-->
<!-- <el-form-item :label="t('填写须知')">-->
<!-- <el-input v-model.trim="formData.write_instruction" :placeholder="t('请输入填写须知')" type="textarea" maxlength="500" show-word-limit rows="5" class="w-[400px]" clearable />-->
<!-- </el-form-item>-->
</el-form>
<template #footer>
<span class="dialog-footer">
<el-button @click="showDialog = false">{{ t('cancel') }}</el-button>
<el-button type="primary" :loading="loading" @click="confirm(formRef)">{{ t('confirm') }}</el-button>
</span>
</template>
</el-dialog>
</template>
<script lang="ts" setup>
import { ref, reactive, computed } from 'vue'
import { t } from '@/lang'
import type { FormInstance } from 'element-plus'
import { filterNumber } from '@/utils/common'
import {getMemberLabelAll,getMemberLevelAll } from '@/app/api/member'
import { getFormWriteConfig,editDiyFormWriteConfig } from '@/app/api/diy_form'
const showDialog = ref(false)
const loading = ref(false)
/**
* 表单数据
*/
const initialFormData = {
id: 0,
form_id: 0, // id
write_way: 'no_limit', // no_limitscanurl
join_member_type: 'all_member', // all_memberselected_member_levelselected_member_label
level_ids: [], // id
label_ids: [], // id
member_write_type: 'no_limit', // no_limitdiy
//
member_write_rule: {
time_value: 1, //
time_unit: 'day', //
num: 1 //
},
form_write_type: 'no_limit', // no_limitdiy
//
form_write_rule: {
time_value: 1, //
time_unit: 'day', //
num: 1 //
},
time_limit_type: 'no_limit', // no_limitspecify_timeopen_day_time
//
time_limit_rule: {
specify_time: [], //
open_day_time: [], //
},
is_allow_update_content: 0, // 01
write_instruction: '', //
}
const formData: Record<string, any> = reactive({ ...initialFormData })
const formRef = ref<FormInstance>()
//
const formRules = computed(() => {
return {
label_ids: [
{ required: true, message: t('labelTips'), trigger: 'blur' }
],
level_ids: [
{ required: true, message: t('levelTips'), trigger: 'blur' }
],
member_write_rule: [
{
validator: (rule: any, value: string, callback: any) => {
let unit = ''
validityOptions.forEach((item,index)=>{
if(item.value == value.time_unit){
unit = item.text;
}
})
if(formData.member_write_type == 'diy'){
if(!value.time_value){
callback(new Error(`${unit}数不能为空`))
}else if(!value.num){
callback(new Error(t('numCannotNull')))
}else{
callback()
}
}else{
callback()
}
},
trigger: ['blur', 'change']
}
],
form_write_rule: [
{
validator: (rule: any, value: string, callback: any) => {
let unit = ''
validityOptions.forEach((item,index)=>{
if(item.value == value.time_unit){
unit = item.text;
}
})
if(formData.member_write_type == 'diy'){
if(!value.time_value){
callback(new Error(`${unit}数不能为空`))
}else if(!value.num){
callback(new Error(t('numCannotNull')))
}else{
callback()
}
}else{
callback()
}
},
trigger: ['blur', 'change']
}
],
time_limit_rule: [
{
validator: (rule: any, value: string, callback: any) => {
if (formData.time_limit_type == 'specify_time' && (!value.specify_time || !value.specify_time.length)) {
callback(new Error(t('timeLimitRuleOne')))
} else if (formData.time_limit_type == 'open_day_time' && (!value.open_day_time || !value.open_day_time.length)) {
callback(new Error(t('timeLimitRuleTwo')))
} else if (formData.time_limit_type == 'open_day_time' && value.open_day_time && value.open_day_time.length) {
if (value.open_day_time[0] == value.open_day_time[1]) {
callback(new Error(t('timeLimitRuleThree')))
} else {
callback()
}
} else {
callback()
}
},
trigger: ['blur', 'change']
}
]
}
})
const levelSelectData = ref([])
const labelSelectData = ref([])
//
getMemberLabelAll().then(({ data }) => {
labelSelectData.value = data
})
getMemberLevelAll().then(({ data }) => {
levelSelectData.value = data
})
const validityOptions = reactive([
{
text:'天',
value:'day'
},
{
text:'周',
value:'week'
},
{
text:'月',
value:'month'
},
{
text:'年',
value:'year'
}
])
const emit = defineEmits(['complete'])
const setFormData = async (row: any = null) => {
Object.assign(formData, initialFormData)
loading.value = true
if (row) {
const data = await (await getFormWriteConfig(row.form_id)).data
if (data && Object.keys(data).length) {
Object.keys(formData).forEach((key: string) => {
if (data[key] != undefined) formData[key] = data[key]
})
}else{
formData.form_id = row.form_id;
}
}
loading.value = false
}
/**
* 确认
* @param formEl
*/
const confirm = async (formEl: FormInstance | undefined) => {
if (loading.value || !formEl) return
await formEl.validate(async (valid) => {
if (valid) {
loading.value = true
const data = formData
editDiyFormWriteConfig(data).then(res => {
loading.value = false
showDialog.value = false
emit('complete')
}).catch(err => {
loading.value = false
})
}
})
}
const filterSpecial = (event:any) => {
event.target.value = event.target.value.replace(/[^\u4e00-\u9fa5a-zA-Z0-9]/g, '')
event.target.value = event.target.value.replace(/[`~!@#$%^&*()_\-+=<>?:"{}|,.\/;'\\[\]·~@#¥%……&*()——\-+={}|《》?:“”【】、;‘’,。、]/g, '')
}
defineExpose({
showDialog,
setFormData
})
</script>
<style lang="scss" scoped></style>
<style lang="scss">
.diy-dialog-wrap .el-form-item__label{
height: auto !important;
}
.display-block {
.el-form-item__content {
display: block;
}
}
</style>

File diff suppressed because it is too large Load Diff

View File

@ -0,0 +1,584 @@
<template>
<div class="main-container">
<el-card class="box-card !border-none" shadow="never">
<div class="flex justify-between items-center">
<span class="text-page-title">{{ pageName }}</span>
<el-button type="primary" class="w-[100px]" @click="dialogVisible = true">{{ t('addDiyForm') }}</el-button>
</div>
<el-card class="box-card !border-none my-[10px] table-search-wrap" shadow="never">
<el-form :inline="true" :model="diyFormTableData.searchParam" ref="searchFormDiyFormRef">
<el-form-item :label="t('title')" prop="title">
<el-input v-model.trim="diyFormTableData.searchParam.title" :placeholder="t('titlePlaceholder')" />
</el-form-item>
<!-- <el-form-item :label="t('forAddon')" prop="addon_name">-->
<!-- <el-select v-model="diyFormTableData.searchParam.addon_name" :placeholder="t('forAddonPlaceholder')" @change="handleSelectAddonChange">-->
<!-- <el-option :label="t('all')" value="" />-->
<!-- <el-option v-for="(item, key) in apps" :label="item.title" :value="key" :key="key"/>-->
<!-- </el-select>-->
<!-- </el-form-item>-->
<el-form-item :label="t('typeName')" prop="type">
<el-select v-model="diyFormTableData.searchParam.type" :placeholder="t('formTypePlaceholder')">
<el-option :label="t('all')" value="" />
<el-option v-for="(item, key) in formType" :label="item.title" :value="key" :key="key"/>
</el-select>
</el-form-item>
<el-form-item>
<el-button type="primary" @click="loadDiyFormList()">{{ t('search') }}</el-button>
<el-button @click="resetForm(searchFormDiyFormRef)">{{ t('reset') }}</el-button>
</el-form-item>
</el-form>
</el-card>
<div class="mb-[10px] flex items-center">
<el-checkbox v-model="toggleCheckbox" size="large" class="px-[14px]" @change="toggleChange" :indeterminate="isIndeterminate" />
<el-button @click="batchDeleteForms" size="small">{{t("batchDeletion")}}</el-button>
</div>
<el-table :data="diyFormTableData.data" size="large" ref="diyFormListTableRef" v-loading="diyFormTableData.loading" @selection-change="handleSelectionChange">
<template #empty>
<span>{{ !diyFormTableData.loading ? t('emptyData') : '' }}</span>
</template>
<el-table-column type="selection" width="55" />
<el-table-column prop="page_title" :label="t('title')" min-width="120" />
<!-- <el-table-column prop="addon_name" :label="t('forAddon')" min-width="80" />-->
<el-table-column prop="type_name" :label="t('typeName')" min-width="80" />
<el-table-column :label="t('status')" min-width="80">
<template #default="{ row }">
<el-tag type="success" v-if="row.status == 1" class="cursor-pointer" @click="showClick(row)">{{ t('statusOn') }}</el-tag>
<el-tag type="info" v-else class="cursor-pointer" @click="showClick(row)">{{ t('statusOff') }}</el-tag>
</template>
</el-table-column>
<el-table-column prop="update_time" :label="t('updateTime')" min-width="120" />
<el-table-column :label="t('operation')" fixed="right" align="right" min-width="100">
<template #default="{ row }">
<div class="flex items-center justify-end">
<el-button type="primary" v-if="row.status == 1 && row.type=='DIY_FORM'" link @click="spreadEvent(row)">{{ t('promotion') }}</el-button>
<!-- <el-button type="primary" link @click="toPreview(row)">{{ t('preview') }}</el-button>-->
<el-button type="primary" link @click="editEvent(row)">{{ t('edit') }}</el-button>
<el-button v-if="row.status == 0" type="primary" link @click="deleteEvent(row.form_id)">{{ t('delete') }}</el-button>
<el-button type="primary" link @click="detailEvent(row)">{{ t('detail') }}</el-button>
<el-dropdown placement="bottom" trigger="click" class="ml-[12px]">
<el-button type="primary" link>{{ t('more') }}</el-button>
<template #dropdown>
<el-dropdown-menu>
<el-dropdown-item v-if="row.type=='DIY_FORM'">
<el-button type="primary" class="w-full" link @click="submitConfigEvent(row)">{{ t('submirSuccess') }}</el-button>
</el-dropdown-item>
<el-dropdown-item>
<el-button type="primary" class="w-full" link @click="writeConfigEvent(row)">{{ t('writeSet') }}</el-button>
</el-dropdown-item>
<el-dropdown-item v-if="row.type=='DIY_FORM'">
<el-button type="primary" class="w-full" link @click="openShare(row)">{{ t('shareSet') }}</el-button>
</el-dropdown-item>
<el-dropdown-item>
<el-button type="primary" class="w-full" link @click="exportEvent(row)">{{ t('export') }}</el-button>
</el-dropdown-item>
<el-dropdown-item>
<el-button type="primary" class="w-full" link @click="copyEvent(row.form_id)">{{ t('copy') }}</el-button>
</el-dropdown-item>
</el-dropdown-menu>
</template>
</el-dropdown>
</div>
</template>
</el-table-column>
</el-table>
<div class="mt-[16px] flex justify-end">
<el-pagination v-model:current-page="diyFormTableData.page" v-model:page-size="diyFormTableData.limit"
layout="total, sizes, prev, pager, next, jumper" :total="diyFormTableData.total"
@size-change="loadDiyFormList()" @current-change="loadDiyFormList" />
</div>
</el-card>
<!--添加表单-->
<el-dialog v-model="dialogVisible" :title="t('addFormTips')" width="980px">
<el-form :model="formData" ref="formRef" :rules="formRules">
<!-- <el-form-item :label="t('title')" prop="title">-->
<!-- <el-input v-model.trim="formData.title" :placeholder="t('titlePlaceholder')" clearable maxlength="12" show-word-limit class="w-full" />-->
<!-- </el-form-item>-->
<el-form-item prop="type">
<div class="image-selection-container">
<div
v-for="(item, key) in formType"
:key="key"
class="image-option"
:class="{ selected: formData.type === key }"
@click="selectType(key)"
>
<img :src="img(item.preview)" class="option-image" />
<div class="option-title">{{ item.title }}</div>
</div>
</div>
<!-- <el-select v-model="formData.type" :placeholder="t('formTypePlaceholder')" class="!w-full">
<el-option v-for="(item, key) in formType" :label="item.title" :value="key" :key="key"/>
</el-select> -->
</el-form-item>
</el-form>
<template #footer>
<span class="dialog-footer">
<el-button @click="dialogVisible = false">{{ t('cancel') }}</el-button>
<el-button type="primary" @click="addEvent(formRef)">{{ t('confirm') }}</el-button>
</span>
</template>
</el-dialog>
<!-- 分享设置-->
<el-dialog v-model="shareDialogVisible" :title="t('shareSet')" width="30%">
<el-tabs v-model="tabShareType">
<el-tab-pane :label="t('wechat')" name="wechat"></el-tab-pane>
<el-tab-pane :label="t('weapp')" name="weapp"></el-tab-pane>
</el-tabs>
<el-form :model="shareFormData[tabShareType]" label-width="90px" ref="shareFormRef" :rules="shareFormRules">
<el-form-item :label="t('sharePage')">
<span>{{ sharePage }}</span>
</el-form-item>
<el-form-item :label="t('shareTitle')" prop="title">
<el-input v-model.trim="shareFormData[tabShareType].title" :placeholder="t('shareTitlePlaceholder')" clearable maxlength="30" show-word-limit />
</el-form-item>
<el-form-item :label="t('shareDesc')" prop="desc" v-if="tabShareType == 'wechat'">
<el-input v-model.trim="shareFormData[tabShareType].desc" :placeholder="t('shareDescPlaceholder')" type="textarea" rows="4" clearable maxlength="100" show-word-limit />
</el-form-item>
<el-form-item :label="t('shareImageUrl')" prop="url">
<upload-image v-model="shareFormData[tabShareType].url" :limit="1" />
</el-form-item>
</el-form>
<template #footer>
<span class="dialog-footer">
<el-button @click="shareDialogVisible = false">{{ t('cancel') }}</el-button>
<el-button type="primary" @click="shareEvent(shareFormRef)">{{ t('confirm') }}</el-button>
</span>
</template>
</el-dialog>
<!-- 推广弹出框 -->
<form-spread-popup ref="formSpreadPopupRef" />
<!-- 表单提交成功页弹出框 -->
<form-submit-popup ref="formSubmitPopupRef" @complete="loadDiyFormList" />
<!-- 表单填写设置弹出框 -->
<form-write-popup ref="formWritePopupRef" @complete="loadDiyFormList" />
<records-detail ref="recordsDetailDialog"/>
<!-- 表单明细导出弹出框 -->
<export-sure ref="exportSureDialog" :show="flag" type="diy_form_records" :searchParam="diyFormDetailData" @close="handleExportClose" />
</div>
</template>
<script lang="ts" setup>
import { reactive, ref, computed } from 'vue'
import { t } from '@/lang'
import { getFormType, getApps, getDiyFormPageList, deleteDiyForm, editDiyFormShare, editFormStatus, copyForm } from '@/app/api/diy_form'
import { FormInstance, ElMessage, ElMessageBox } from "element-plus";
import { useRoute, useRouter } from 'vue-router'
import { setTablePageStorage,getTablePageStorage } from "@/utils/common";
import { img } from '@/utils/common'
import recordsDetail from '@/app/views/diy_form/records.vue'
import formSpreadPopup from '@/app/views/diy_form/components/form-spread-popup.vue'
import formSubmitPopup from '@/app/views/diy_form/components/form-submit-popup.vue'
import formWritePopup from '@/app/views/diy_form/components/form-write-popup.vue'
const route = useRoute()
const router = useRouter()
const pageName = route.meta.title
const repeat = ref(false)
const formType: any = reactive({}) //
//
const formData = reactive({
title: '',
type: ''
})
//
const recordsDetailDialog: Record<string, any> | null = ref(null)
const detailEvent=(row: any)=>{
let data = {form_id: row.form_id};
recordsDetailDialog.value.setFormData(data);
recordsDetailDialog.value.showDialog = true;
}
//
const formRules = computed(() => {
return {
title: [
{ required: true, message: t('titlePlaceholder'), trigger: 'blur' }
],
type: [
{ required: true, message: t('formTypePlaceholder'), trigger: 'blur' }
]
}
})
const formRef = ref<FormInstance>()
const dialogVisible = ref(false)
const addEvent = async (formEl: FormInstance | undefined) => {
if (!formEl) return
await formEl.validate(async(valid) => {
if (valid) {
const query = { type: formData.type } // , title: formData.title
const url = router.resolve({
path: '/decorate/form/edit',
query
})
window.open(url.href)
dialogVisible.value = false
formData.title = ''
formData.type = ''
}
})
}
const showClick = (row: any) => {
const data = row.status === 1 ? 0 : 1
const obj = {
form_id: row.form_id,
status: data
}
editFormStatus(obj).then((res) => {
row.status = data
})
}
//
const loadFormType = (addon = '')=> {
getFormType({}).then(res => {
for (let key in formType) {
delete formType[key];
}
for (const key in res.data) {
formType[key] = res.data[key]
}
formData.type = Object.keys(formType)[0]
})
}
loadFormType();
const apps: any = reactive({}) //
// todo
// getApps({}).then(res=>{
// if(res.data){
// for (const key in res.data) {
// apps[key] = res.data[key];
// }
// }
// });
//
const handleSelectAddonChange = (value: any) => {
diyFormTableData.searchParam.type = '';
loadFormType(value)
}
const diyFormTableData: any = reactive({
page: 1,
limit: 10,
total: 0,
loading: true,
data: [],
searchParam: {
title: '',
type: '',
mode: '',
addon_name: ''
}
})
const searchFormDiyFormRef = ref<FormInstance>()
//
const loadDiyFormList = (page: number = 1) => {
diyFormTableData.loading = true
diyFormTableData.page = page
getDiyFormPageList({
page: diyFormTableData.page,
limit: diyFormTableData.limit,
...diyFormTableData.searchParam
}).then(res => {
diyFormTableData.loading = false
diyFormTableData.data = res.data.data
diyFormTableData.total = res.data.total
setTablePageStorage(diyFormTableData.page, diyFormTableData.limit, diyFormTableData.searchParam);
}).catch(() => {
diyFormTableData.loading = false
})
}
loadDiyFormList(getTablePageStorage(diyFormTableData.searchParam).page);
const selectType = (index: number) => {
formData.type = index.toString();
}
const resetForm = (formEl: FormInstance | undefined) => {
if (!formEl) return
formEl.resetFields()
loadDiyFormList()
}
//
const editEvent = (data: any) => {
const url = router.resolve({
path: '/decorate/form/edit',
query: { form_id: data.form_id }
})
window.open(url.href)
}
//
const copyEvent = (id: any) => {
ElMessageBox.confirm(t('diyFormCopyTips'), t('warning'),
{
confirmButtonText: t('confirm'),
cancelButtonText: t('cancel'),
type: 'warning'
}
).then(() => {
if (repeat.value) return
repeat.value = true
copyForm({form_id: id}).then((res: any) => {
if (res.code == 1) {
loadDiyFormList()
}
repeat.value = false
}).catch(() => {
repeat.value = false
})
})
}
//
const deleteEvent = (form_id: number) => {
ElMessageBox.confirm(t('diyFormDeleteTips'), t('warning'),
{
confirmButtonText: t('confirm'),
cancelButtonText: t('cancel'),
type: 'warning'
}
).then(() => {
deleteDiyForm({ form_ids: [form_id] }).then(() => {
loadDiyFormList()
}).catch(() => {
})
})
}
//
const toggleCheckbox = ref();
//
const isIndeterminate = ref(false);
//
const toggleChange = (value: any) => {
isIndeterminate.value = false;
diyFormListTableRef.value.toggleAllSelection();
};
const diyFormListTableRef = ref();
//
const multipleSelection: any = ref([]);
//
const handleSelectionChange = (val: []) => {
multipleSelection.value = val;
toggleCheckbox.value = false;
if (
multipleSelection.value.length > 0 &&
multipleSelection.value.length < diyFormTableData.data.length
) {
isIndeterminate.value = true;
} else {
isIndeterminate.value = false;
}
if (multipleSelection.value.length == diyFormTableData.data.length) {
toggleCheckbox.value = true;
}
};
//
const batchDeleteForms = () => {
if (multipleSelection.value.length == 0) {
ElMessage({
type: "warning",
message: `${t("batchEmptySelectedFormsTips")}`,
});
return;
}
ElMessageBox.confirm(t("batchFormsDeleteTips"), t("warning"), {
confirmButtonText: t("confirm"),
cancelButtonText: t("cancel"),
type: "warning",
}).then(() => {
if (repeat.value) return;
repeat.value = true;
const form_ids: any = [];
multipleSelection.value.forEach((item: any) => {
form_ids.push(item.form_id);
});
deleteDiyForm({
form_ids: form_ids,
}).then(() => {
loadDiyFormList();
repeat.value = false;
}).catch(() => {
repeat.value = false;
});
});
};
//
const toPreview = (data: any) => {
const url = router.resolve({
path: '/preview/wap',
query: {
page: '/app/pages/index/diy_form?form_id=' + data.form_id
}
});
window.open(url.href)
}
const tabShareType = ref('wechat')
const sharePage = ref('')
const shareFormId = ref(0)
const shareFormData = reactive({
wechat: {
title: '',
desc: '',
url: ''
},
weapp: {
title: '',
url: ''
}
})
const shareDialogVisible = ref(false)
const shareFormRules = computed(() => {
return {}
})
const shareFormRef = ref<FormInstance>()
const openShare = async (row: any) => {
shareFormId.value = row.form_id
sharePage.value = row.title
const share = row.share
shareFormData.wechat = share.wechat
shareFormData.weapp = share.weapp
shareDialogVisible.value = true
}
const shareEvent = async (formEl: FormInstance | undefined) => {
if (!formEl) return
await formEl.validate(async (valid) => {
if (valid) {
editDiyFormShare({
form_id: shareFormId.value,
share: JSON.stringify(shareFormData)
}).then(() => {
loadDiyFormList()
shareDialogVisible.value = false
}).catch(() => {
})
}
})
}
// 广
const formSpreadPopupRef: any = ref(null)
const spreadEvent = (data: any) => {
formSpreadPopupRef.value.show(data)
}
//
const formSubmitPopupRef: any = ref(null)
const submitConfigEvent = (data: any) => {
formSubmitPopupRef.value.setFormData(data)
formSubmitPopupRef.value.showDialog = true
}
//
const formWritePopupRef: any = ref(null)
const writeConfigEvent = (data: any) => {
formWritePopupRef.value.setFormData(data)
formWritePopupRef.value.showDialog = true
}
/**
* 表单填写记录明细导出
*/
const exportSureDialog = ref(null)
const flag = ref(false)
const handleExportClose = (val) => {
flag.value = val
}
const diyFormDetailData: any = reactive({
form_id: 0,
})
const exportEvent = (data: any) => {
diyFormDetailData.form_id = data.form_id
flag.value = true
}
</script>
<style lang="scss" scoped>
.image-selection-container {
width: 100%;
display: flex;
gap: 20px;
justify-content: center;
}
.image-option {
cursor: pointer;
display: flex;
flex-direction: column;
align-items: center;
width: 300px;
border: 2px solid transparent;
border-radius: 10px;
transition: border-color 0.3s ease, box-shadow 0.3s ease;
overflow: hidden;
}
.image-option.selected {
border-color: var(--el-color-primary);
box-shadow: 0 0 10px var(--el-color-primary-light-9);
}
.option-image {
width: 100%;
height: auto;
object-fit: cover;
}
.option-title {
margin-top: 10px;
font-size: 14px;
color: #303133;
font-weight: bold;
text-align: center;
}
</style>

View File

@ -0,0 +1,391 @@
<template>
<el-drawer v-model="showDialog" :title="t('dataAndStatistics')" direction="rtl" size="70%" :before-close="handleClose" class="member-detail-drawer">
<el-tabs v-model="activeName" class="demo-tabs" @tab-click="handleClick">
<el-tab-pane :label="t('detailData')" name="detail_data">
<el-card class="box-card !border-none my-[10px] table-search-wrap" shadow="never">
<el-form :inline="true" :model="formData.searchParam" ref="searchFormDiyFormRef">
<el-form-item :label="t('fillInFormPerson')" prop="keyword">
<el-input v-model.trim="formData.searchParam.keyword" :placeholder="t('fillInFormPersonplaceholder')" />
</el-form-item>
<el-form-item :label="t('fillInFormDate')" prop="create_time">
<el-date-picker v-model="formData.searchParam.create_time" type="datetimerange" value-format="YYYY-MM-DD HH:mm:ss" :start-placeholder="t('startDate')" :end-placeholder="t('endDate')" />
</el-form-item>
<el-form-item>
<el-button type="primary" @click="loadFormRecordsListFn()">{{ t('search') }}</el-button>
<el-button @click="resetForm(searchFormDiyFormRef)">{{ t('reset') }}</el-button>
<el-button type="primary" @click="exportEvent">{{ t('export') }}</el-button>
</el-form-item>
</el-form>
</el-card>
<el-table :data="formData.data" size="large" v-loading="formData.loading">
<template #empty>
<span>{{ !formData.loading ? t('emptyData') : '' }}</span>
</template>
<el-table-column fixed :label="t('fillInFormPersonInfo')" min-width="160">
<template #default="{ row }">
<div class="flex items-center cursor-pointer" @click="detailEvent(row.member.member_id)">
<div class="min-w-[50px] h-[50px] flex items-center justify-center">
<el-image v-if="row.member.headimg" class="w-[50px] h-[50px]" :src="img(row.member.headimg)" fit="contain">
<template #error>
<div class="image-slot">
<img class="w-[50px] h-[50px] rounded-full" src="@/app/assets/images/member_head.png" alt="">
</div>
</template>
</el-image>
<img class="w-[50px] h-[50px] rounded-full" v-else src="@/app/assets/images/member_head.png" alt="">
</div>
<div class="ml-2">
<span :title="(row.member.nickname || row.member.username)" class="multi-hidden">{{row.member.nickname || row.member.username}}</span>
<span class="text-primary text-[12px]">{{row.member.mobile || ''}}</span>
</div>
</div>
</template>
</el-table-column>
<el-table-column fixed prop="create_time" :label="t('fillInFormDate')" min-width="120" />
<el-table-column v-for="item in formFieldsList" :key="item.field_key" :label="item.field_name" min-width="200">
<template #default="{ row }">
<!-- 动态渲染表单组件内容 -->
<template v-if="row.recordsFieldList[item.field_key]">
<component :is="row.recordsFieldList[item.field_key].detailComponent" :data="row.recordsFieldList[item.field_key]"/>
</template>
</template>
</el-table-column>
<el-table-column :label="t('operation')" fixed="right" align="right" min-width="70">
<template #default="{ row }">
<!-- <el-button type="primary" link @click="formDetailEvent(row)">{{ t('详情') }}</el-button> -->
<el-button type="primary" link @click="deleteEvent(row)">{{ t('delete') }}</el-button>
</template>
</el-table-column>
</el-table>
<div class="mt-[16px] flex justify-end">
<el-pagination v-model:current-page="formData.page" v-model:page-size="formData.limit" :page-sizes="[6,10,20,30,50,100]"
layout="total, sizes, prev, pager, next, jumper" :total="formData.total"
@size-change="loadFormRecordsListFn()" @current-change="loadFormRecordsListFn" />
</div>
</el-tab-pane>
<el-tab-pane :label="t('fillInFormPersonStatics')" name="member_stat">
<el-card class="box-card !border-none my-[10px] table-search-wrap" shadow="never">
<el-form :inline="true" :model="formMemberList.searchParam" ref="searchFormDiyMemberRef">
<el-form-item :label="t('fillInFormPerson')" prop="keyword">
<el-input v-model.trim="formMemberList.searchParam.keyword" :placeholder="t('fillInFormPersonplaceholder')" />
</el-form-item>
<el-form-item>
<el-button type="primary" @click="getFormRecordsMemberFn()">{{ t('search') }}</el-button>
<el-button @click="resetFormMember(searchFormDiyMemberRef)">{{ t('reset') }}</el-button>
<el-button type="primary" @click="exportMemberEvent">{{ t('export') }}</el-button>
</el-form-item>
</el-form>
</el-card>
<el-table :data="formMemberList.data" size="large" v-loading="formMemberList.loading">
<template #empty>
<span>{{ !formMemberList.loading ? t('emptyData') : '' }}</span>
</template>
<el-table-column fixed :label="t('fillInFormPersonInfo')" min-width="200">
<template #default="{ row }">
<div class="flex items-center cursor-pointer" @click="detailEvent(row.member.member_id)">
<div class="min-w-[50px] h-[50px] flex items-center justify-center">
<el-image v-if="row.member.headimg" class="w-[50px] h-[50px]" :src="img(row.member.headimg)" fit="contain">
<template #error>
<div class="image-slot">
<img class="w-[50px] h-[50px] rounded-full" src="@/app/assets/images/member_head.png" alt="">
</div>
</template>
</el-image>
<img class="w-[50px] h-[50px] rounded-full" v-else src="@/app/assets/images/member_head.png" alt="">
</div>
<div class="ml-2">
<span :title="(row.member.nickname || row.member.username)" class="multi-hidden">{{row.member.nickname || row.member.username}}</span>
<span class="text-primary text-[12px]">{{row.member.mobile || ''}}</span>
</div>
</div>
</template>
</el-table-column>
<!-- <el-table-column fixed prop="create_time" :label="t('填表时间')" min-width="120" /> -->
<el-table-column fixed prop="create_time" :label="t('fillInFormTotal')" min-width="500">
<template #default="{ row }" @click="">
{{ row.write_count }}
</template>
</el-table-column>
</el-table>
<div class="mt-[16px] flex justify-end">
<el-pagination v-model:current-page="formMemberList.page" v-model:page-size="formMemberList.limit"
layout="total, sizes, prev, pager, next, jumper" :total="formMemberList.total"
@size-change="getFormRecordsMemberFn()" @current-change="getFormRecordsMemberFn()" />
</div>
</el-tab-pane>
<el-tab-pane :label="t('fieldStatistics')" name="field_stat">
<el-card class="box-card !border-none my-[10px] table-search-wrap" shadow="never">
<el-form :inline="true" :model="formData.searchParam" ref="searchFormDiyFieldsRef">
<el-form-item>
<el-button type="primary" @click="exportFieldsEvent">{{ t('export') }}</el-button>
</el-form-item>
</el-form>
</el-card>
<el-collapse v-model="activeNames" class="diy-collapse mt-[15px]">
<el-collapse-item :title="item.field_name" :name="item.field_id" v-for="(item, index) in formFieldsStat" :key="index">
<template #title>
<div class="text-[16px] font-bold">{{item.field_name}}</div>
</template>
<el-table :data="item.value_list" border>
<el-table-column :label="item.field_name" prop="render_value">
<template #default="{ row }">
{{row.render_value ? row.render_value : '未填写'}}
</template>
</el-table-column>
<el-table-column label="小计" prop="write_count"></el-table-column>
<el-table-column label="比例">
<template #default="{ row }">
<el-progress :percentage="row.write_percent"></el-progress>
</template>
</el-table-column>
</el-table>
</el-collapse-item>
</el-collapse>
</el-tab-pane>
</el-tabs>
<el-dialog v-model="dialogVisible" :title="t('viewInformation')" width="400px">
<div class="flex flex-col">
<div class="flex mb-[10px]" v-for="(item, index) in formDetail" :key="index">
<div class="flex justify-end w-[100px]">{{ item.label }}</div>
<div class="flex ml-[20px]">
<div v-if="Array.isArray(item.text)" class="mr-[10px]" v-for="(textItem, i) in item.text" :key="i">
{{ textItem }}
</div>
<div v-else>{{ item.text }}</div>
</div>
</div>
</div>
<template #footer>
<span class="dialog-footer">
<el-button type="primary" @click="dialogVisible = false">{{ t('confirm') }}</el-button>
</span>
</template>
</el-dialog>
</el-drawer>
<export-sure ref="exportSureDialog" :show="flag" type="diy_form_records" :searchParam="formData.searchParam" @close="handleExportClose" />
<export-sure ref="exportSureDialog" :show="flagMember" type="diy_form_records_member" :searchParam="formMemberList.searchParam" @close="handleMemberExportClose" />
<export-sure ref="exportSureDialog" :show="flagFields" type="diy_form_records_fields" :searchParam="formData.searchParam" @close="handleFieldsExportClose" />
</template>
<script lang="ts" setup>
import { reactive, ref, defineAsyncComponent } from 'vue'
import { t } from '@/lang'
import { getDiyFormFieldsList, getDiyFormFieldStat, getFormRecords,getFormRecordsInfo,deleteFormRecords,getFormRecordsMember} from '@/app/api/diy_form'
import { useRouter, useRoute } from 'vue-router'
import { img } from '@/utils/common'
import { ElMessageBox, FormInstance } from 'element-plus'
const route = useRoute()
const router = useRouter()
const showDialog = ref(false)
const activeName = ref('detail_data')
const formId = ref(0)
const dialogVisible = ref(false)
const searchFormDiyFormRef = ref<FormInstance>()
const searchFormDiyMemberRef = ref<FormInstance>()
const searchFormDiyFieldsRef = ref<FormInstance>()
const handleClose = (done: () => void) => {
showDialog.value = false;
}
const formData = reactive({
page: 1,
limit: 6,
total: 0,
loading: false,
data: [],
searchParam: {
form_id: 0,
keyword: '',
create_time: ''
}
})
const formFieldsList = ref([])
//
const getDiyFormFieldsListFn = (form_id:any)=>{
getDiyFormFieldsList({
form_id,
order: 'field_id',
sort: 'asc'
}).then((res:any)=>{
formFieldsList.value = res.data;
})
}
//
const formFieldsStat = ref([])
const getDiyFormFieldStatFn = (form_id:any)=>{
getDiyFormFieldStat({
form_id
}).then((res:any)=>{
formFieldsStat.value = res.data;
})
}
const modules: any = import.meta.glob('@/**/*.vue')
const formDetail = ref([])
const formDetailEvent = (row: any) => {
getFormRecordsInfo(row.record_id).then((res:any)=>{
formDetail.value = res.data.value
dialogVisible.value = true
})
}
//
const deleteEvent = (row: any) => {
ElMessageBox.confirm(t('deleteTips'), t('warning'),
{
confirmButtonText: t('confirm'),
cancelButtonText: t('cancel'),
type: 'warning'
}
).then(() => {
deleteFormRecords({
record_id: row.record_id,
form_id: row.form_id
}).then(() => {
initData();
}).catch(() => {
})
})
}
const resetForm = (formEl: FormInstance | undefined) => {
if (!formEl) return
formEl.resetFields()
loadFormRecordsListFn()
}
const resetFormMember = (formEl: FormInstance | undefined) => {
if (!formEl) return
formEl.resetFields()
getFormRecordsMemberFn()
}
const loadFormRecordsListFn= (page: number = 1)=>{
formData.loading = true
formData.page = page
getFormRecords({
page: formData.page,
limit: formData.limit,
...formData.searchParam
}).then((res:any)=>{
formData.loading = false
formData.data = res.data.data
formData.data.forEach((item:any)=>{
for (let key:any in item.recordsFieldList){
if (modules[item.recordsFieldList[key].detailComponent]) {
item.recordsFieldList[key].detailComponent && (item.recordsFieldList[key].detailComponent = defineAsyncComponent(modules[item.recordsFieldList[key].detailComponent]))
}
}
})
formData.total = res.data.total
}).catch(() => {
formData.loading = false
})
}
const formMemberList = reactive({
page: 1,
limit: 10,
total: 0,
loading: false,
data: [],
searchParam: {
keyword: '',
form_id: 0,
}
})
const getFormRecordsMemberFn = (page: number = 1) => {
formMemberList.loading = true
formMemberList.page = page
getFormRecordsMember({
page: formMemberList.page,
limit: formMemberList.limit,
...formMemberList.searchParam
}).then((res: any) => {
formMemberList.data = res.data.data;
formMemberList.total = res.total;
formMemberList.loading = false;
}).catch((error) => {
formMemberList.loading = false;
});
}
//
const detailEvent = (member_id:number)=> {
let routeData = router.resolve(`/member/detail?id=${member_id}`)
window.open(routeData.href, ' blank');
}
const setFormData = async (row: any = null) => {
formId.value = row.form_id;
formData.searchParam.form_id = row.form_id;
formMemberList.searchParam.form_id = row.form_id;
getDiyFormFieldsListFn(row.form_id);
initData();
}
const initData = () =>{
getFormRecordsMemberFn();
getDiyFormFieldStatFn(formId.value);
loadFormRecordsListFn()
}
/**
* 表单填写记录明细导出
*/
const exportSureDialog = ref(null)
const flag = ref(false)
const handleExportClose = (val) => {
flag.value = val
}
const exportEvent = () => {
flag.value = true
}
const flagMember = ref(false)
const handleMemberExportClose = (val) => {
flagMember.value = val
}
const exportMemberEvent = () => {
flagMember.value = true
}
const flagFields = ref(false)
const handleFieldsExportClose = (val) => {
flagFields.value = val
}
const exportFieldsEvent = () => {
flagFields.value = true
}
defineExpose({
showDialog,
setFormData
})
</script>
<style lang="scss">
.diy-collapse .el-collapse-item__header{
display: flex;
flex-direction: row-reverse;
justify-content: flex-end;
.el-icon.el-collapse-item__arrow{
margin-left: inherit;
margin-right: 10px;
}
}
</style>

View File

@ -3,7 +3,7 @@
<div class="flex items-center">
<slot name="content">
<div>
<img class="w-[300px]" src="@/app/assets/images/error.png" />
<img class="w-[240px]" src="@/app/assets/images/error.png" />
</div>
</slot>
<div class="text-left ml-[100px]">

View File

@ -44,8 +44,8 @@
:placeholder="t('cashOutNumberPlaceholder')" />
</el-form-item>
<el-form-item :label="t('memberInfo')" prop="keyword">
<el-input v-model.trim="orderTableData.searchParam.keyword" class="w-[240px]"
<el-form-item :label="t('memberInfo')" prop="keywords">
<el-input v-model.trim="orderTableData.searchParam.keywords" class="w-[240px]"
:placeholder="t('memberInfoPlaceholder')" />
</el-form-item>
@ -80,64 +80,109 @@
</el-card>
<div class="mt-[10px]">
<el-table :data="orderTableData.data" size="large" v-loading="orderTableData.loading">
<template #empty>
<span>{{ !orderTableData.loading ? t('emptyData') : '' }}</span>
</template>
<el-table :data="orderTableData.data" size="large" class="table-top">
<el-table-column :label="t('memberInfo')" min-width="180" />
<el-table-column :label="t('cashOutMethod')" align="center" min-width="100" />
<el-table-column :label="t('cashOutInfo')" min-width="180" />
<el-table-column :label="t('applicationForWithdrawalAmount')" align="center" min-width="120" />
<el-table-column :label="t('actualTransferAmount')" align="center" min-width="120" />
<el-table-column :label="t('cashOutCommission')" align="center" min-width="110" />
<el-table-column :label="t('cashOutStatus')" align="center" min-width="100" />
<el-table-column :label="t('applyTime')" align="center" min-width="160" />
<el-table-column :label="t('auditTime')" align="center" min-width="160" />
<el-table-column :label="t('transferTime')" align="center" min-width="160" />
<el-table-column :label="t('operation')" fixed="right" align="right" min-width="120" />
</el-table>
<div class="table-body min-h-[150px]" v-loading="orderTableData.loading">
<div v-if="!orderTableData.loading">
<template v-if="orderTableData.data.length">
<div v-for="(item, index) in orderTableData.data" :key="index">
<el-table :data="[item]" size="large" :show-header="false">
<el-table-column :show-overflow-tooltip="true" min-width="180">
<template #default="{ row }">
<div class="flex items-center cursor-pointer " @click="toMember(row.member.member_id)">
<img class="w-[50px] h-[50px] mr-[10px]" v-if="row.member.headimg" :src="img(row.member.headimg)" alt="">
<img class="w-[50px] h-[50px] mr-[10px] rounded-full" v-else src="@/app/assets/images/member_head.png" alt="">
<div class="flex flex flex-col items-baseline" style="width: calc(100% - 60px);">
<span class="w-[100%] truncate text-left">{{ row.member.nickname || row.member.username || '' }}</span>
<span class="w-[100%] truncate">{{ row.member.mobile || '' }}</span>
</div>
</div>
</template>
</el-table-column>
<el-table-column align="center" min-width="100">
<template #default="{ row }">
{{ row.transfer_type_name }}
</template>
</el-table-column>
<el-table-column min-width="180">
<template #default="{ row }">
<div class="flex flex-col" v-if="row.transfer_type=='wechat_code' || row.transfer_type=='alipay'">
<div class="flex items-center">
<span class="w-[70px] flex-shrink-0 text-right">{{t('realname') }}</span>
<span class="using-hidden">{{ row.transfer_realname }}</span>
</div>
<div class="flex items-center">
<span class="w-[70px] flex-shrink-0 text-right">{{t('account') }}</span>
<span>{{ row.transfer_account }}</span>
</div>
<div class="flex items-center" v-if="row.transfer_payment_code">
<span class="w-[70px] flex-shrink-0 text-right">{{ t('transferCode') }}</span>
<el-image :src="img(row.transfer_payment_code)" :preview-src-list="[img(row.transfer_payment_code)]" :hide-on-click-modal="true" class="w-[50px] h-[50px]"></el-image>
</div>
</div>
<div class="flex flex-col" v-else-if="row.transfer_type=='bank'">
<span>{{t('bankRealname') }}{{ row.transfer_realname }}</span>
<span>{{ t('bankAccount') }}{{ row.transfer_account }}</span>
<span>{{ t('bankName') }}{{ row.transfer_bank }}</span>
</div>
</template>
</el-table-column>
<el-table-column prop="apply_money" min-width="120" align="center" />
<el-table-column prop="order_no" :show-overflow-tooltip="true" :label="t('memberInfo')" align="center" min-width="140">
<template #default="{ row }">
<div class="flex items-center cursor-pointer " @click="toMember(row.member.member_id)">
<img class="w-[50px] h-[50px] mr-[10px]" v-if="row.member.headimg" :src="img(row.member.headimg)" alt="">
<img class="w-[50px] h-[50px] mr-[10px] rounded-full" v-else src="@/app/assets/images/member_head.png" alt="">
<div class="flex flex flex-col">
<span>{{ row.member.nickname || '' }}</span>
<span>{{ row.member.mobile || '' }}</span>
</div>
<el-table-column prop="money" min-width="120" align="center" />
<el-table-column prop="service_money" align="center" min-width="110" />
<el-table-column prop="status_name" align="center" min-width="100" />
<el-table-column min-width="160" align="center">
<template #default="{ row }">
{{ row.create_time || '' }}
</template>
</el-table-column>
<el-table-column min-width="160" align="center">
<template #default="{ row }">
{{ row.audit_time || '' }}
</template>
</el-table-column>
<el-table-column min-width="160" align="center">
<template #default="{ row }">
{{ row.transfer_time || '' }}
</template>
</el-table-column>
<el-table-column align="right" fixed="right" width="120">
<template #default="{ row }">
<el-button v-for="(item, index) in operationBtn[row.status.toString()].value" :key="index + 'a'"
@click="fnProcessing(operationBtn[row.status.toString()].clickArr[index], row)"
type="primary" link>{{ item }}</el-button>
<el-button type="primary" link @click="handleRemark(row)"> {{ t('remark') }}</el-button>
</template>
</el-table-column>
</el-table>
<div v-if="item.remark" class="text-[14px] min-h-[30px] leading-[30px] px-3 bg-[#fff0e5] text-[#ff7f5b] mb-[10px] relative remark">
<span class="mr-[5px]">{{ t('notes') }}</span>
<span>{{ item.remark }}</span>
</div>
</div>
</template>
</el-table-column>
<el-table-column :label="t('cashOutMethod')" align="center" min-width="140">
<template #default="{ row }">
{{ row.transfer_type_name }}
</template>
</el-table-column>
<el-empty v-else :image-size="1" :description="t('emptyData')" />
</div>
</div>
<el-table-column prop="apply_money" :label="t('applicationForWithdrawalAmount')" min-width="140" align="center" />
<el-table-column prop="money" :label="t('actualTransferAmount')" min-width="200" align="center" />
<el-table-column prop="service_money" :label="t('cashOutCommission')" align="center" min-width="140" />
<el-table-column prop="status_name" :label="t('cashOutStatus')" align="center" min-width="100" />
<el-table-column :label="t('applyTime')" min-width="180" align="center">
<template #default="{ row }">
{{ row.create_time || '' }}
</template>
</el-table-column>
<el-table-column :label="t('auditTime')" min-width="180" align="center">
<template #default="{ row }">
{{ row.audit_time || '' }}
</template>
</el-table-column>
<el-table-column :label="t('transferTime')" min-width="180" align="center">
<template #default="{ row }">
{{ row.transfer_time || '' }}
</template>
</el-table-column>
<el-table-column :label="t('operation')" align="right" fixed="right" width="120">
<template #default="{ row }">
<el-button v-for="(item, index) in operationBtn[row.status.toString()].value" :key="index + 'a'"
@click="fnProcessing(operationBtn[row.status.toString()].clickArr[index], row)"
type="primary" link>{{ item }}</el-button>
</template>
</el-table-column>
</el-table>
<div class="mt-[16px] flex justify-end">
<el-pagination v-model:current-page="orderTableData.page" v-model:page-size="orderTableData.limit"
layout="total, sizes, prev, pager, next, jumper" :total="orderTableData.total"
@ -147,42 +192,99 @@
</el-card>
<!-- 详情 -->
<el-dialog v-model="cashOutShowDialog" :title="t('cashOutDetail')" width="500px" :destroy-on-close="true">
<el-form :model="cashOutInfo" label-width="120px" ref="formRef" :rules="formRules" class="page-form" v-loading="cashOutLoading">
<el-form-item :label="t('nickname')">
<div class="input-width"> {{ cashOutInfo.nickname }} </div>
</el-form-item>
<el-form-item :label="t('cashOutAccountType')">
<div class="input-width"> {{ cashOutInfo.account_type_name }} </div>
</el-form-item>
<el-form-item :label="t('cashOutMethod')">
<div class="input-width"> {{ Transfertype[cashOutInfo.transfer_type].name }} </div>
</el-form-item>
<template v-if="cashOutInfo.transfer_type == 'alipay'">
<el-form-item :label="t('alipayAccount')">
<div class="input-width"> {{ cashOutInfo.transfer_account }} </div>
</el-form-item>
</template>
<template v-if="cashOutInfo.transfer_type == 'bank'">
<el-form-item :label="t('bankName')">
<div class="input-width"> {{ cashOutInfo.transfer_bank }} </div>
</el-form-item>
<el-form-item :label="t('bankAccount')">
<div class="input-width"> {{ cashOutInfo.transfer_account }} </div>
</el-form-item>
</template>
<el-form-item :label="t('applicationForWithdrawalAmount')">
<div class="input-width"> {{ cashOutInfo.apply_money }} </div>
</el-form-item>
<el-form-item :label="t('cashOutCommission')">
<div class="input-width"> {{ cashOutInfo.service_money }} </div>
</el-form-item>
<el-form-item :label="t('actualTransferAmount')">
<div class="input-width"> {{ cashOutInfo.money }} </div>
</el-form-item>
<el-form-item :label="t('cashOutStatus')">
<div class="input-width"> {{ cashOutInfo.status_name }} </div>
</el-form-item>
<el-dialog v-model="cashOutShowDialog" :title="t('cashOutDetail')" width="650px" :destroy-on-close="true">
<el-form :model="cashOutInfo" label-width="120px" ref="formRef" class="page-form" v-loading="cashOutLoading">
<el-row>
<el-col :span="12">
<el-form-item :label="t('nickname')">
<div class="input-width"> {{ cashOutInfo.nickname|| cashOutInfo.username }} </div>
</el-form-item>
</el-col>
<el-col :span="12">
<el-form-item :label="t('cashOutAccountType')">
<div class="input-width"> {{ cashOutInfo.account_type_name }} </div>
</el-form-item>
</el-col>
<el-col :span="12">
<el-form-item :label="t('cashOutMethod')">
<div class="input-width"> {{ Transfertype[cashOutInfo.transfer_type].name }} </div>
</el-form-item>
</el-col>
<template v-if="cashOutInfo.transfer_type == 'alipay' || cashOutInfo.transfer_type == 'wechat_code'">
<el-col :span="12">
<el-form-item :label="t('realname')">
<div class="input-width"> {{ cashOutInfo.transfer_realname }} </div>
</el-form-item>
</el-col>
<el-col :span="12">
<el-form-item :label="t('alipayAccount')">
<div class="input-width"> {{ cashOutInfo.transfer_account }} </div>
</el-form-item>
</el-col>
<el-col :span="12">
<el-form-item :label="t('transferCode')">
<el-image :src="img(cashOutInfo.transfer_payment_code)" :preview-src-list="[img(cashOutInfo.transfer_payment_code)]" :hide-on-click-modal="true" class="mr-[10px] w-[50px] h-[50px]"></el-image>
</el-form-item>
</el-col>
</template>
<template v-if="cashOutInfo.transfer_type == 'bank'">
<el-col :span="12">
<el-form-item :label="t('bankName')">
<div class="input-width"> {{ cashOutInfo.transfer_bank }} </div>
</el-form-item>
</el-col>
<el-col :span="12">
<el-form-item :label="t('bankAccount')">
<div class="input-width"> {{ cashOutInfo.transfer_account }} </div>
</el-form-item>
</el-col>
</template>
<el-col :span="12">
<el-form-item :label="t('applicationForWithdrawalAmount')">
<div class="input-width"> {{ cashOutInfo.apply_money }} </div>
</el-form-item>
</el-col>
<el-col :span="12">
<el-form-item :label="t('cashOutCommission')">
<div class="input-width"> {{ cashOutInfo.service_money }} </div>
</el-form-item>
</el-col>
<el-col :span="12">
<el-form-item :label="t('actualTransferAmount')">
<div class="input-width"> {{ cashOutInfo.money }} </div>
</el-form-item>
</el-col>
<el-col :span="12">
<el-form-item :label="t('cashOutStatus')">
<div class="input-width"> {{ cashOutInfo.status_name }} </div>
</el-form-item>
</el-col>
<el-col :span="12">
<el-form-item :label="t('applyTime')">
<div class="input-width"> {{ cashOutInfo.create_time }} </div>
</el-form-item>
</el-col>
<el-col :span="12">
<el-form-item :label="t('auditTime')">
<div class="input-width"> {{ cashOutInfo.audit_time }} </div>
</el-form-item>
</el-col>
<el-col :span="12" v-if="cashOutInfo.remark">
<el-form-item :label="t('remark')">
<div class="input-width"> {{ cashOutInfo.remark }} </div>
</el-form-item>
</el-col>
<el-col :span="12" v-if="cashOutInfo.transfer && cashOutInfo.transfer.transfer_voucher">
<el-form-item :label="t('transferVoucher')">
<el-image :src="img(cashOutInfo.transfer.transfer_voucher)" :preview-src-list="[img(cashOutInfo.transfer.transfer_voucher)]" :hide-on-click-modal="true" class="w-[50px] h-[50px]"></el-image>
</el-form-item>
</el-col>
<el-col :span="12" v-if="cashOutInfo.transfer && cashOutInfo.transfer.transfer_remark">
<el-form-item :label="t('transferRemark')">
<div class="input-width"> {{ cashOutInfo.transfer.transfer_remark }} </div>
</el-form-item>
</el-col>
</el-row>
</el-form>
<template #footer>
@ -191,11 +293,92 @@
</span>
</template>
</el-dialog>
<!-- 是否审核 -->
<!-- 审核通过 -->
<el-dialog v-model="auditPassShowDialog" :title="t('passAudit')" width="650px" :destroy-on-close="true">
<el-form :model="curData" label-width="120px" ref="formRef" class="page-form">
<el-row>
<el-col :span="12">
<el-form-item :label="t('nickname')">
<div class="input-width"> {{ curData.member.nickname ||curData.member.username }} </div>
</el-form-item>
</el-col>
<el-col :span="12">
<el-form-item :label="t('cashOutAccountType')">
<div class="input-width"> {{ curData.account_type_name }} </div>
</el-form-item>
</el-col>
<el-col :span="12">
<el-form-item :label="t('cashOutMethod')">
<div class="input-width"> {{ curData.transfer_type_name }} </div>
</el-form-item>
</el-col>
<template v-if="curData.transfer_type == 'alipay' || curData.transfer_type == 'wechat_code'">
<el-col :span="12">
<el-form-item :label="t('realname')">
<div class="input-width"> {{ curData.transfer_realname }} </div>
</el-form-item>
</el-col>
<el-col :span="12">
<el-form-item :label="t('account')">
<div class="input-width"> {{ curData.transfer_account }} </div>
</el-form-item>
</el-col>
<el-col :span="12">
<el-form-item :label="t('transferCode')">
<el-image :src="img(curData.transfer_payment_code)" :preview-src-list="[img(curData.transfer_payment_code)]" :hide-on-click-modal="true" class="w-[50px] h-[50px]"></el-image>
</el-form-item>
</el-col>
</template>
<template v-if="curData.transfer_type == 'bank'">
<el-col :span="12">
<el-form-item :label="t('bankName')">
<div class="input-width"> {{ curData.transfer_bank }} </div>
</el-form-item>
</el-col>
<el-col :span="12">
<el-form-item :label="t('bankRealname')">
<div class="input-width"> {{ curData.transfer_realname }} </div>
</el-form-item>
</el-col>
<el-col :span="12">
<el-form-item :label="t('bankAccount')">
<div class="input-width"> {{ curData.transfer_account }} </div>
</el-form-item>
</el-col>
</template>
<el-col :span="12">
<el-form-item :label="t('applicationForWithdrawalAmount')">
<div class="input-width"> {{ curData.apply_money }} </div>
</el-form-item>
</el-col>
<el-col :span="12">
<el-form-item :label="t('cashOutCommission')">
<div class="input-width"> {{ curData.service_money }} </div>
</el-form-item>
</el-col>
<el-col :span="12">
<el-form-item :label="t('actualTransferAmount')">
<div class="input-width"> {{ curData.money }} </div>
</el-form-item>
</el-col>
<el-col :span="12">
<el-form-item :label="t('applyTime')">
<div class="input-width"> {{ curData.create_time }} </div>
</el-form-item>
</el-col>
</el-row>
</el-form>
<template #footer>
<span class="dialog-footer">
<el-button @click="auditPassShowDialog = false">{{ t('cancel') }}</el-button>
<el-button type="primary" @click="handlePass()">{{ t('confirm') }}</el-button>
</span>
</template>
</el-dialog>
<!-- 是否审核拒绝 -->
<el-dialog v-model="auditShowDialog" :title="t('rejectionAudit')" width="500px" :destroy-on-close="true">
<el-form :model="auditFailure" label-width="90px" ref="formRef" :rules="formRules" class="page-form" v-loading="loading">
<el-form-item :label="t('reasonsRefusal')" prop="label_name">
<el-form-item :label="t('reasonsRefusal')" prop="refuse_reason">
<el-input v-model.trim="auditFailure.refuse_reason" clearable maxlength="200" :show-word-limit="true" :placeholder="t('reasonsRefusalPlaceholder')" :rows="4" class="input-width" type="textarea" />
</el-form-item>
</el-form>
@ -208,12 +391,88 @@
</el-dialog>
<!-- 是否转账 -->
<el-dialog v-model="transferShowDialog" :title="t('rejectionAudit')" width="500px" :destroy-on-close="true">
<p>{{ t('isTransfer') }}</p>
<el-dialog v-model="transferShowDialog" :title="t('transfer')" width="650px" :destroy-on-close="true">
<el-form :model="transferData" label-width="120px" ref="formRef" class="page-form">
<el-row>
<template v-if="transferData.transfer_type == 'alipay' || transferData.transfer_type == 'wechat_code'">
<el-col :span="12">
<el-form-item :label="t('realname')">
<div class="input-width"> {{ transferData.transfer_realname }} </div>
</el-form-item>
</el-col>
<el-col :span="12">
<el-form-item :label="t('account')">
<div class="input-width"> {{ transferData.transfer_account }} </div>
</el-form-item>
</el-col>
<el-col :span="12">
<el-form-item :label="t('transferCode')">
<el-image :src="img(transferData.transfer_payment_code)" :preview-src-list="[img(transferData.transfer_payment_code)]" :hide-on-click-modal="true" class="w-[50px] h-[50px]"></el-image>
</el-form-item>
</el-col>
</template>
<template v-if="transferData.transfer_type == 'bank'">
<el-col :span="12">
<el-form-item :label="t('bankName')">
<div class="input-width"> {{ transferData.transfer_bank }} </div>
</el-form-item>
</el-col>
<el-col :span="12">
<el-form-item :label="t('bankRealname')">
<div class="input-width"> {{ transferData.transfer_realname }} </div>
</el-form-item>
</el-col>
<el-col :span="12">
<el-form-item :label="t('bankAccount')">
<div class="input-width"> {{ transferData.transfer_account }} </div>
</el-form-item>
</el-col>
</template>
<el-col :span="12">
<el-form-item :label="t('applicationForWithdrawalAmount')">
<div class="input-width"> {{ transferData.apply_money }} </div>
</el-form-item>
</el-col>
<el-col :span="12">
<el-form-item :label="t('cashOutCommission')">
<div class="input-width"> {{ transferData.service_money }} </div>
</el-form-item>
</el-col>
<el-col :span="12">
<el-form-item :label="t('actualTransferAmount')">
<div class="input-width"> {{ transferData.money }} </div>
</el-form-item>
</el-col>
</el-row>
</el-form>
<el-form :model="formTransfer" label-width="120px" ref="formTransferRef" :rules="formTransferRules" class="page-form">
<el-form-item :label="t('transferVoucher')" prop="transfer_voucher">
<upload-image v-model="formTransfer.transfer_voucher" :limit="1" />
</el-form-item>
<el-form-item :label="t('transferRemark')" prop="transfer_remark">
<el-input v-model.trim="formTransfer.transfer_remark" type="textarea" rows="4" clearable
:placeholder="t('transferRemarkPlaceholder')" class="input-width" maxlength="200" show-word-limit />
</el-form-item>
</el-form>
<template #footer>
<span class="dialog-footer">
<el-button @click="transferShowDialog = false">{{ t('cancel') }}</el-button>
<el-button type="primary" @click="confirm()">{{ t('confirm') }}</el-button>
<el-button type="primary" @click="handleTransfer(formTransferRef)">{{ t('confirm') }}</el-button>
</span>
</template>
</el-dialog>
<!-- 备注 -->
<el-dialog v-model="remarkShowDialog" :title="t('remark')" width="500px" :destroy-on-close="true">
<el-form :model="formData" label-width="90px" ref="formRemarkRef" :rules="formRemarkRules" class="page-form">
<el-form-item :label="t('remark')" prop="remark">
<el-input v-model.trim="formData.remark" type="textarea" rows="4" clearable
:placeholder="t('remarkPlaceholder')" class="input-width" maxlength="200" show-word-limit />
</el-form-item>
</el-form>
<template #footer>
<span class="dialog-footer">
<el-button @click="remarkShowDialog = false">{{ t('cancel') }}</el-button>
<el-button type="primary" @click="save(formRemarkRef)">{{ t('confirm') }}</el-button>
</span>
</template>
</el-dialog>
@ -221,9 +480,9 @@
</template>
<script lang="ts" setup>
import { reactive, ref } from 'vue'
import { reactive, ref, computed } from 'vue'
import { t } from '@/lang'
import { getCashOutList, getTransfertype, memberTransfer, memberAudit, getCashOutDetail, getCashOutStatusList, getCashOutStat } from '@/app/api/member'
import { getCashOutList, getTransfertype, memberTransfer, memberAudit, getCashOutDetail, getCashOutStatusList, getCashOutStat, memberRemark, memberCheck } from '@/app/api/member'
import { img } from '@/utils/common'
import { ElMessageBox, FormInstance, FormRules } from 'element-plus'
import { useRouter, useRoute } from 'vue-router'
@ -234,7 +493,6 @@ const checkStatusList = async () => {
cashOutStatusList.value = await (await getCashOutStatusList()).data
}
checkStatusList()
const transferShowDialog = ref(false)
const route = useRoute()
const router = useRouter()
const pageName = route.meta.title
@ -251,6 +509,10 @@ const operationBtn = ref<AnyObject>({
value: [t('detail')],
clickArr: ['detailFn']
},
4: {
value: [t('detail')],
clickArr: ['detailFn']
},
'-1': {
value: [t('detail')],
clickArr: ['detailFn']
@ -276,7 +538,7 @@ const orderTableData = reactive({
create_time: [],
status: '',
cash_out_no: '',
keyword: '',
keywords: '',
audit_time: '',
transfer_time: '',
transfer_type: ''
@ -339,17 +601,26 @@ const fnProcessing = (type: string, data: any) => {
obj.id = data.id
if (type == 'successfulAuditFn') {
obj.action = 'agree'
cashOutAuditFn(obj)
curData.value = data
auditPassShowDialog.value = true
} else {
obj.action = 'refuse'
auditFailure.value = Object.assign(auditFailure.value, obj)
auditShowDialog.value = true
}
} else if (type == 'transferFn') {
obj.id = data.id
ElMessageBox.confirm(`${ t('isTransfer') }`, `${ t('transfer') }`).then(() => {
transferFn(obj)
})
if (data.transfer_type == 'wechatpay') {
obj.id = data.id
ElMessageBox.confirm(`${t('isTransfer')}`, `${t('transfer')}`).then(() => {
transferFn(obj)
})
} else {
transferData.value = data
formTransfer.id = data.id
transferShowDialog.value = true
}
} else if (type == 'checkFn') {
checkFn(data.id)
} else {
detailFn(data.id)
}
@ -359,12 +630,36 @@ const fnProcessing = (type: string, data: any) => {
* 转账
* @param data
*/
const transferData = ref({})
const transferShowDialog = ref(false)
const formTransferRef = ref<FormInstance>()
const formTransfer = reactive({
id: 0,
transfer_voucher: '',
transfer_remark: ''
})
const formTransferRules = computed(() => {
return {
transfer_voucher: [
{ required: true, message: t('transferVoucherPlaceholder'), trigger: 'blur' }
]
}
})
const handleTransfer = async (formEl: FormInstance | undefined) => {
if (!formEl) return
await formEl.validate(async (valid) => {
if (valid) {
transferFn(formTransfer)
}
})
}
const transferFn = (data:any) => {
memberTransfer({ ...data }).then(res => {
auditFailure.value = { refuse_reason: '', id: 0, action: 0 }
transferShowDialog.value = false
loadOrderList()
}).catch(() => {
transferShowDialog.value = false
loadOrderList()
})
}
@ -398,13 +693,24 @@ const detailFn = (id:any) => {
* 提现审核
* @param data
*/
const auditPassShowDialog = ref(false)
const curData = ref<any>({})
const handlePass = () => {
const obj = {
id: curData.value.id,
action: 'agree'
}
cashOutAuditFn(obj)
}
const cashOutAuditFn = (data:any) => {
memberAudit({
...data
}).then(res => {
auditPassShowDialog.value = false
loadOrderList()
}).catch(() => {
auditPassShowDialog.value = false
loadOrderList()
})
}
@ -417,6 +723,41 @@ const confirm = () => {
cashOutAuditFn(auditFailure.value)
}
/**
* 备注
*/
const formRemarkRef = ref<FormInstance>()
const remarkShowDialog = ref(false)
const formData = reactive({
id: 0,
remark: ''
})
const formRemarkRules = computed(() => {
return {
remark: [
{ required: true, message: t('remarkPlaceholder'), trigger: 'blur' }
]
}
})
const handleRemark = (data: any) => {
formData.id = data.id
formData.remark = ''
remarkShowDialog.value = true
}
const save = async (formEl: FormInstance | undefined) => {
if (!formEl) return
await formEl.validate(async (valid) => {
if (valid) {
memberRemark(formData).then((res: any) => {
loadOrderList()
remarkShowDialog.value = false
}).catch(() => {
remarkShowDialog.value = false
})
}
})
}
/**
* 会员详情
*/
@ -424,6 +765,30 @@ const toMember = (memberId: number) => {
router.push(`/member/detail?id=${memberId}`)
}
//
const checkFn = (id: number) => {
memberCheck(id).then(res => {
})
}
</script>
<style lang="scss" scoped></style>
<style lang="scss" scoped>
.table-top :deep(.el-table__body-wrapper) {
display: none;
}
:deep(.el-table) {
--el-table-row-hover-bg-color: var(--el-transfer-border-color);
}
.remark{
&::after{
content: '';
border-bottom: solid 1px var(--el-border-color-light);
width: 100%;
position: absolute;
bottom: -10px;
left: 0;
}
}
</style>

View File

@ -1,7 +1,7 @@
<template>
<el-form :model="formData" :rules="formRules" class="page-form" ref="formRef">
<el-form-item :label="t('continueSign')" prop="continue_sign">
<el-input class="input-width" v-model.trim="formData.continue_sign" :maxlength="5" clearable />
<el-input class="input-width" v-model.trim="formData.continue_sign" @keyup="filterNumber($event)" :maxlength="3" clearable />
<span class="ml-[10px]">{{ t('day') }}</span>
</el-form-item>
<el-form-item :label="t('continueSign')" >
@ -28,7 +28,7 @@ import { t } from '@/lang'
import { ref, reactive, defineAsyncComponent, computed, watch } from 'vue'
import { FormRules } from 'element-plus'
import { getGiftDict } from '@/app/api/member'
import { guid } from '@/utils/common'
import { guid, filterNumber } from '@/utils/common'
import Test from '@/utils/test'
const gifts = ref({})
@ -88,13 +88,12 @@ const regExp = {
const formRules = reactive<FormRules>({
continue_sign: [
{ required: true, message: t('continueSignPlaceholder'), trigger: 'blur' },
{
{
validator: (rule: any, value: any, callback: any) => {
if (isNaN(value) || !regExp.number.test(value)) {
callback('连续签到天数格式错误')
} else if (value <=0) {
callback('连续签到天数不能小于等于0')
callback(t('continueSignFormatError'))
} else if (value < 2 || value > 365) {
callback(t('continueSignBerweenDays'))
} else{
callback();
}
@ -108,13 +107,13 @@ const formRules = reactive<FormRules>({
validator: (rule: any, value: any, callback: Function) => {
if (formData.value.receive_limit == 2) {
if (Test.empty(formData.value.receive_num)) {
callback('请输入限领次数')
callback(t('receiveNumPlaceholder'))
}
if (isNaN(formData.value.receive_num) || !regExp.number.test(formData.value.receive_num)) {
callback('限领次数格式错误')
callback(t('receiveNumFormatError'))
}
if (formData.value.receive_num <= 0) {
callback('限领次数不能小于等于0')
callback(t('receiveNumMustGreaterThanZeroTip'))
}
callback()
} else {

View File

@ -11,7 +11,7 @@
</el-form-item>
<el-form-item :label="t('signPeriod')" prop="sign_period" v-if="formData.is_use">
<el-input v-model="formData.sign_period" placeholder="0" maxlength="8" clearable class="input-width" /><span class="ml-[10px]"></span>
<el-input v-model.trim="formData.sign_period" @keyup="filterNumber($event)" maxlength="3" clearable class="input-width" /><span class="ml-[10px]"></span>
</el-form-item>
<el-form-item :label="t('daySignAward')" prop="day_award" v-if="formData.is_use">
@ -115,6 +115,7 @@ import { getSignConfig, setSignConfig, getMemberGiftsContent } from '@/app/api/m
import signDay from '@/app/views/marketing/components/sign-day.vue'
import signContinue from '@/app/views/marketing/components/sign-continue.vue'
import { FormInstance, FormRules } from 'element-plus'
import { filterNumber } from '@/utils/common'
import { useRoute } from 'vue-router'
import { cloneDeep } from 'lodash-es'
@ -150,7 +151,7 @@ const formRules = reactive<FormRules>({
callback(t('signPeriodTip'))
}else if (isNaN(value) || !regExp.number.test(value)) {
callback(t('signPeriodLimitTips'))
}else if (value <= 0) {
}else if (value < 2 || value > 365) {
callback(t('signPeriodMustZeroTips'))
} else {
callback();

View File

@ -28,7 +28,7 @@
</el-select>
</el-form-item>
<el-form-item :label="t('memberLevel')" prop="member_label">
<el-form-item :label="t('memberLevel')" prop="member_level">
<el-select v-model="memberTableData.searchParam.member_level" collapse-tags clearable :placeholder="t('memberLevelPlaceholder')" class="input-width">
<el-option :label="t('selectPlaceholder')" value="" />
<el-option :label="item['level_name']" :value="item['level_id']" v-for="(item, index) in levelSelectData" :key="index" />

View File

@ -49,7 +49,7 @@
<el-form-item :label="t('printWidth')" prop="print_width">
<el-radio-group v-model="formData.print_width">
<el-radio value="58mm">58mm</el-radio>
<el-radio label="58mm">58mm</el-radio>
</el-radio-group>
</el-form-item>

View File

@ -15,7 +15,10 @@
</el-form-item>
<el-form-item :label="t('cashWithdrawalAmount')" v-if="formData.is_open" prop="min">
<el-input v-model.trim="formData.min" @keyup="filterDigit($event)" class="input-width" :placeholder="t('cashWithdrawalAmountPlaceholder')" />
<div>
<el-input v-model.trim="formData.min" @keyup="filterDigit($event)" class="input-width" :placeholder="t('cashWithdrawalAmountPlaceholder')" />
<div class="text-[12px] text-[#999] leading-[24px]">{{ t('minTips') }}</div>
</div>
</el-form-item>
<el-form-item :label="t('commissionRatio')" v-if="formData.is_open" prop="rate">
@ -30,17 +33,22 @@
</el-radio-group>
</el-form-item>
<el-form-item :label="t('transfer')" v-if="formData.is_open" class="items-center">
<el-radio-group v-model="formData.is_auto_transfer">
<el-radio label="0" size="large">{{t('manualTransfer')}}</el-radio>
<el-radio label="1" size="large">{{t('automatedTransit')}}</el-radio>
</el-radio-group>
<el-form-item :label="t('transfer')" v-if="formData.is_open" class="items-baseline">
<div>
<el-radio-group v-model="formData.is_auto_transfer">
<el-radio label="0" size="large">{{t('manualTransfer')}}</el-radio>
<el-radio label="1" size="large">{{t('automatedTransit')}}</el-radio>
</el-radio-group>
<div class="text-[12px] text-[#999] leading-[24px]">{{ t('transferTips') }}</div>
</div>
</el-form-item>
<el-form-item :label="t('transferMode')" v-if="formData.is_open" class="items-center">
<el-checkbox-group v-model="formData.transfer_type" size="large">
<el-checkbox :label="item.key" v-for="(item,index) in Transfertype" :key="'a'+index">{{item.name}}</el-checkbox>
</el-checkbox-group>
<el-form-item :label="t('transferMode')" v-if="formData.is_open" class="items-baseline">
<div>
<el-checkbox-group v-model="formData.transfer_type" size="large">
<el-checkbox :label="item.key" v-for="(item,index) in Transfertype" :key="'a'+index">{{item.name}}</el-checkbox>
</el-checkbox-group>
<div class="text-[12px] text-[#999] leading-[24px]">{{ t('transferModeTips') }}</div>
</div>
</el-form-item>
</el-card>
</el-form>

View File

@ -7,19 +7,23 @@
</el-form-item>
<el-form-item :label="t('smsType')">
<div class="input-width"> {{ formData.sms_type_name }} </div>
<div class="input-width">
<div v-if="formData.notice_type == 'sms'">{{ t('sms') }}</div>
<div v-if="formData.notice_type == 'wechat'">{{ t('wechat') }}</div>
<div v-if="formData.notice_type == 'weapp'">{{ t('weapp') }}</div>
</div>
</el-form-item>
<!-- <el-form-item :label="t('messageData')">
<div class="input-width"> {{ formData.message_data }} </div>
</el-form-item> -->
<!-- <el-form-item :label="t('nickname')">
<el-form-item :label="t('nickname')">
<div class="input-width"> {{ formData.nickname }} </div>
</el-form-item> -->
</el-form-item>
<el-form-item :label="t('receiver')">
<div class="input-width"> {{ formData.mobile }} </div>
<div class="input-width"> {{ formData.receiver }} </div>
</el-form-item>
<el-form-item :label="t('createTime')">
@ -54,8 +58,8 @@ const initialFormData = {
message_type: '',
name: '',
nickname: '',
mobile: '',
sms_type_name: ''
receiver: '',
notice_type: ''
}
const formData: Record<string, any> = reactive({ ...initialFormData })

View File

@ -30,6 +30,12 @@
<el-form-item :label="t('remark')" prop="config.pay_leave_message">
<el-input v-model.trim="formData.config.pay_leave_message" :placeholder="t('remarkPlaceholder')" class="input-width" type="textarea" rows="4" maxlength="20" show-word-limit clearable />
</el-form-item>
<el-form-item :label="t('payWechatImage')" prop="config.pay_wechat_share_image" v-if="initData.redio_key == 'wechat_friendspay'">
<upload-image v-model="formData.config.pay_wechat_share_image" :limit="1" />
</el-form-item>
<el-form-item :label="t('payWeappImage')" prop="config.pay_weapp_share_image" v-if="initData.redio_key == 'weapp_friendspay'">
<upload-image v-model="formData.config.pay_weapp_share_image" :limit="1" />
</el-form-item>
</el-form>
<template #footer>
<span class="dialog-footer">
@ -64,6 +70,8 @@ const initialFormData = {
pay_page_name: '',
pay_button_name: '',
pay_leave_message: '',
pay_wechat_share_image: '',
pay_weapp_share_image: ''
},
channel: '',
status: 0,
@ -146,6 +154,7 @@ const cancel = () => {
}
const setFormData = async (data: any = null) => {
console.log(data)
initData.value = cloneDeep(data)
loading.value = true
Object.assign(formData, initialFormData)

View File

@ -10,12 +10,11 @@
<el-card class="box-card !border-none my-[10px] table-search-wrap" shadow="never">
<el-form :inline="true" :model="recordsTableData.searchParam" ref="searchFormRef">
<el-form-item :label="t('searchReceiver')" prop="receiver">
<el-input v-model="recordsTableData.searchParam.receiver" :placeholder="t('receiverPlaceholder')" />
<el-input v-model.trim="recordsTableData.searchParam.receiver" :placeholder="t('receiverPlaceholder')" />
</el-form-item>
<el-form-item :label="t('noticeKey')" prop="key">
<el-select v-model="recordsTableData.searchParam.key" clearable :placeholder="t('noticeKeyPlaceholder')" class="input-width" popper-class="notice">
<el-option :label="t('selectPlaceholder')" value="" />
<el-option-group v-for="(group, gindex) in templateList" :key="gindex" :label="group.label">
<el-option :label="item.name" :value="item.value" :disabled="item.disabled ?? false" v-for="(item, index) in group.list" :key="index" />
</el-option-group>
@ -28,7 +27,7 @@
<el-form-item>
<el-button type="primary" @click="loadNoticeLogList()">{{ t('search') }}</el-button>
<el-button @click="searchFormRef?.resetFields()">{{ t('reset') }}</el-button>
<el-button @click="resetForm(searchFormRef)">{{ t('reset') }}</el-button>
</el-form-item>
</el-form>
</el-card>
@ -159,6 +158,12 @@ const loadNoticeLogList = (page: number = 1) => {
}
loadNoticeLogList()
const resetForm = (formEl: FormInstance | undefined) => {
if (!formEl) return
formEl.resetFields()
loadNoticeLogList()
}
const recordsDialog: Record<string, any> | null = ref(null)
/**

View File

@ -8,7 +8,11 @@
</div>
<div class="mt-[20px]">
<el-alert :title="t('operationTip')" type="warning" show-icon />
<el-alert :description="t('transferTips')" type="warning" show-icon>
<template #title>
<span class="!text-[14px]">{{ t('operationTip') }}</span>
</template>
</el-alert>
</div>
</el-card>
@ -38,7 +42,7 @@
</el-form-item>
</el-card>
<el-card class="box-card mt-[15px] !border-none" shadow="never">
<!-- <el-card class="box-card mt-[15px] !border-none" shadow="never">
<h3 class="panel-title !text-sm">{{t('alipay')}}</h3>
<el-form-item :label="t('appId')" prop="alipay_config.app_id">
@ -63,7 +67,7 @@
<upload-file v-model="formData.alipay_config.alipay_root_cert_path" api="sys/document/aliyun" />
</div>
</el-form-item>
</el-card>
</el-card> -->
</el-form>
<div class="fixed-footer-wrap">