Merge remote-tracking branch 'origin/feature/v1.3_营销资产中台' into feature/v1.3_营销资产中台

This commit is contained in:
林志军
2025-07-10 14:24:11 +08:00
35 changed files with 525 additions and 252 deletions

View File

@ -7,7 +7,11 @@
<script setup lang="ts">
import { useUserStore } from '@/stores';
import { getUserEnterpriseInfo } from '@/utils/user';
import zhCN from '@arco-design/web-vue/es/locale/lang/zh-cn';
const userStore = useUserStore();
const redTheme = {
@ -16,11 +20,14 @@ const redTheme = {
colorLink: '#f5222d', // 链接色
},
};
// 初始化企业信息
const init = async () => {
const { isLogin, getUserInfo } = userStore;
const init = () => {
const { isLogin, fetchUserInfo } = userStore;
if (isLogin) {
fetchUserInfo();
await getUserInfo(); // 初始化用户信息
await getUserEnterpriseInfo();
}
};

View File

@ -1,8 +1,8 @@
/*
* @Author: 田鑫
* @Date: 2023-02-17 11:58:44
* @LastEditors: Please set LastEditors
* @LastEditTime: 2025-07-05 17:59:59
* @LastEditors: rd 1344903914@qq.com
* @LastEditTime: 2025-07-08 14:50:57
* @Description:
*/
@ -42,13 +42,12 @@ export class Request {
this.instance.interceptors.request.use(
(config: AxiosRequestConfig) => {
const store = useEnterpriseStore(pinia);
const enterprise = store.getEnterpriseInfo();
const token = localStorage.getItem('accessToken') as string;
config.headers!.Authorization = token;
if (enterprise) {
config.headers!['enterprise-id'] = enterprise.id;
if (store.enterpriseInfo) {
config.headers!['enterprise-id'] = store.enterpriseInfo.id;
}
return config;

Binary file not shown.

Binary file not shown.

Binary file not shown.

View File

@ -67,7 +67,8 @@ export default defineComponent({
};
listenerRouteChange((newRoute) => {
const { requiresAuth, activeMenu, hideInMenu } = newRoute.meta;
if (requiresAuth && (!hideInMenu || activeMenu)) {
// if (requiresAuth && (!hideInMenu || activeMenu)) {
if (!hideInMenu || activeMenu) {
const menuOpenKeys = findMenuOpenKeys((activeMenu || newRoute.name) as string);
const keySet = new Set([...menuOpenKeys, ...openKeys.value]);
openKeys.value = [...keySet];

View File

@ -12,7 +12,7 @@ export default function useMenuTree() {
// const appStore = useAppStore();
const sidebarStore = useSidebarStore();
const appRoute = computed(() => {
const _filterRoutes = appRoutes.filter((v) => v.meta.id === sidebarStore.activeMenuId);
const _filterRoutes = appRoutes.filter((v) => v.meta?.id === sidebarStore.activeMenuId);
return _filterRoutes;
});
const menuTree = computed(() => {

View File

@ -1,88 +1,42 @@
<script lang="ts" setup>
<script setup>
import { useAppStore } from '@/stores';
import { IconExport, IconFile, IconCaretDown } from '@arco-design/web-vue/es/icon';
import { fetchMenusTree } from '@/api/all';
import { handleUserLogout } from '@/utils/user';
import { fetchLogOut } from '@/api/all/login';
import { useEnterpriseStore } from '@/stores/modules/enterprise';
// import { IconExport, IconFile, IconCaretDown } from '@arco-design/web-vue/es/icon';
// import { handleUserLogout } from '@/utils/user';
// import { fetchLogOut } from '@/api/all/login';
import { useSidebarStore } from '@/stores/modules/side-bar';
import { MENU_GROUP_IDS } from '@/router/constants';
import router from '@/router';
import { useRoute } from 'vue-router';
// import { useRoute } from 'vue-router';
import ExitAccountModal from '@/components/_base/exit-account-modal/index.vue';
import { appRoutes } from '@/router/routes';
// import { appRoutes } from '@/router/routes';
// import { MENU_LIST } from './constants';
interface MenuItem {
name: string;
id: number;
children: Array<{
name: string;
}>;
}
const lists = ref<MenuItem[]>([]);
const sidebarStore = useSidebarStore();
const route = useRoute();
// const enterpriseStore = useEnterpriseStore();
// const route = useRoute();
const exitAccountModalRef = ref(null);
// const selectedKey = ref([]);
const selectedKey = computed(() => {
// 判断是否为工作台页面(假设路由名为 'Home' 或 path 为 '/'
if (route.name === 'Home' || route.path === '/') {
return [`${MENU_GROUP_IDS.WORK_BENCH_ID}`];
}
// 其他页面activeMenuId 作为 key
return [String(sidebarStore.activeMenuId)];
});
const menuList = computed(() => {
return sidebarStore.menuList;
});
const clickExit = async () => {
exitAccountModalRef.value?.open();
};
const getMenus = async () => {
const res = await fetchMenusTree();
if (res.code === 200) {
lists.value = res.data;
}
};
onMounted(() => {
getMenus();
});
const appStore = useAppStore();
// const appStore = useAppStore();
const setServerMenu = () => {
router.push('/management/person');
};
const handleSelect = (index: any) => {
if (index === 0) {
router.push('/');
} else {
router.push('/dataEngine/hotTranslation');
}
};
const flattenRoutes = (routes: any, parentPath = ''): any[] => {
let result: any[] = [];
for (const route of routes) {
const fullPath = route.path.startsWith('/') ? route.path : parentPath.replace(/\/$/, '') + '/' + route.path;
if (route.children && route.children.length) {
result = result.concat(flattenRoutes(route.children, fullPath));
} else {
result.push({ ...route, fullPath });
}
}
return result;
};
const handleDopdownClick = (index: any, ind: any) => {
const { children } = lists.value[index];
const indPath = children[ind];
const allChildren = flattenRoutes(appRoutes);
const target = allChildren.find((item) => item.meta && item.meta.menuId === indPath.id);
if (target) {
router.push(target.fullPath);
} else {
console.warn('未找到对应的菜单路由', indPath.id);
}
const handleDopdownClick = (item) => {
router.push({ name: item.routeName });
};
</script>
@ -95,26 +49,26 @@ const handleDopdownClick = (index: any, ind: any) => {
</div>
<div class="center-side">
<div class="menu-demo h-100%">
<a-menu
mode="horizontal"
:selected-keys="selectedKey"
:default-selected-keys="[`${MENU_GROUP_IDS.WORK_BENCH_ID}`]"
>
<a-menu-item :key="`${MENU_GROUP_IDS.WORK_BENCH_ID}`" @click="handleSelect(0)">
<span class="menu-item-text">工作台</span>
</a-menu-item>
<a-menu-item v-for="(item, index) in lists" :key="String(item.id)">
<a-dropdown :popup-max-height="false" class="layout-menu-item-dropdown">
<a-button>
<span class="menu-item-text mr-2px"> {{ item.name }}</span>
<icon-caret-down size="16" class="arco-icon-down !mr-0" />
</a-button>
<template #content>
<a-doption v-for="(child, ind) in item.children" :key="ind" @click="handleDopdownClick(index, ind)">
<span class="menu-item-text"> {{ child.name }}</span>
</a-doption>
</template>
</a-dropdown>
<a-menu mode="horizontal" :selected-keys="selectedKey">
<a-menu-item v-for="item in menuList" :key="String(item.id)">
<template v-if="item.children">
<a-dropdown :popup-max-height="false" class="layout-menu-item-dropdown">
<a-button>
<span class="menu-item-text mr-2px"> {{ item.name }}</span>
<icon-caret-down size="16" class="arco-icon-down !mr-0" />
</a-button>
<template #content>
<a-doption v-for="(child, ind) in item.children" :key="ind" @click="handleDopdownClick(child)">
<span class="menu-item-text"> {{ child.name }}</span>
</a-doption>
</template>
</a-dropdown>
</template>
<template v-else>
<a-menu-item :key="String(item.id)" @click="handleDopdownClick(item)">
<span class="menu-item-text"> {{ item.name }}</span>
</a-menu-item>
</template>
</a-menu-item>
</a-menu>
</div>

View File

@ -0,0 +1,10 @@
import { useUserStore } from '@/stores/modules/user';
export function checkRoutePermission(routeName: string) {
const userStore = useUserStore();
const allowAccessRoutes = userStore.allowAccessRoutes;
if (!routeName) return false;
return allowAccessRoutes.includes(routeName);
}

View File

@ -5,6 +5,8 @@
import type { Router } from 'vue-router';
import NProgress from 'nprogress';
import { goUserLogin } from '@/utils/user';
// import router from '@/router';
import { checkRoutePermission } from '@/permission/permission';
import { useUserStore } from '@/stores/modules/user';
@ -13,15 +15,27 @@ export default function setupUserLoginInfoGuard(router: Router) {
NProgress.start();
const userStore = useUserStore();
const routeName = to?.name as string;
const requiresAuth = to?.meta?.requiresAuth || false;
const isLogin = !!userStore.isLogin;
const requireLogin = to?.meta?.requireLogin || false;
if (requiresAuth && !isLogin) {
if (requireLogin && !userStore.isLogin) {
goUserLogin();
next();
return;
}
if (requiresAuth) {
const hasPermission = checkRoutePermission(routeName);
if (!hasPermission) {
AMessage.error('您没有权限访问该页面');
next('/');
return;
}
next();
return;
}
next();
});
router.afterEach((to) => {

View File

@ -4,15 +4,15 @@
*/
import { createRouter, createWebHistory } from 'vue-router';
import { appRoutes } from './routes';
import { REDIRECT_MAIN, NOT_FOUND_ROUTE } from './routes/base';
import { NOT_FOUND_ROUTE } from './routes/base';
import NProgress from 'nprogress';
import 'nprogress/nprogress.css';
import { MENU_GROUP_IDS } from './constants';
import createRouteGuard from './guard';
NProgress.configure({ showSpinner: false }); // NProgress Configuration
// console.log({ appRoutes });
const router = createRouter({
export const router = createRouter({
history: createWebHistory(import.meta.env.BASE_URL),
routes: [
{
@ -21,6 +21,7 @@ const router = createRouter({
component: () => import('@/views/components/login'),
meta: {
requiresAuth: false,
requireLogin: false,
},
},
{
@ -29,27 +30,31 @@ const router = createRouter({
component: () => import('@/views/components/workplace'),
meta: {
hideSidebar: true,
requiresAuth: true,
requiresAuth: false,
requireLogin: true,
id: MENU_GROUP_IDS.WORK_BENCH_ID,
},
},
{
path: '/permission',
name: 'permission',
component: () => import('@/views/components/permission/choose-enterprise.vue'),
meta: {
requiresAuth: true,
},
},
{
path: '/auth',
name: 'auth',
component: () => import('@/views/components/permission/auth.vue'),
meta: {
requiresAuth: false,
requireLogin: true,
},
},
// {
// path: '/auth',
// name: 'auth',
// component: () => import('@/views/components/permission/auth.vue'),
// meta: {
// requiresAuth: false,
// requireLogin: true,
// },
// },
...appRoutes,
REDIRECT_MAIN,
// REDIRECT_MAIN,
NOT_FOUND_ROUTE,
],
scrollBehavior() {

View File

@ -1,28 +1,35 @@
import type { RouteRecordRaw } from 'vue-router';
import { REDIRECT_ROUTE_NAME } from '@/router/constants';
export const REDIRECT_MAIN: RouteRecordRaw = {
path: '/redirect',
name: 'redirect',
meta: {
requiresAuth: true,
hideInMenu: true,
},
children: [
{
path: '/redirect/:path',
name: REDIRECT_ROUTE_NAME,
component: () => import('@/layouts/Basic.vue'),
meta: {
requiresAuth: true,
hideInMenu: true,
},
},
],
};
// export const REDIRECT_MAIN: RouteRecordRaw = {
// path: '/redirect',
// name: 'redirect',
// meta: {
// requiresAuth: false,
// requireLogin: false,
// hideInMenu: true,
// },
// children: [
// {
// path: '/redirect/:path',
// name: REDIRECT_ROUTE_NAME,
// component: () => import('@/layouts/Basic.vue'),
// meta: {
// requiresAuth: false,
// requireLogin: false,
// hideInMenu: true,
// },
// },
// ],
// };
export const NOT_FOUND_ROUTE: RouteRecordRaw = {
path: '/:pathMatch(.*)*',
name: 'notFound',
component: () => import('@/layouts/NotFound.vue'),
meta: {
requiresAuth: false,
hideInMenu: true,
hideSidebar: true,
},
};

View File

@ -1,4 +1,6 @@
import type { RouteRecordNormalized } from 'vue-router';
// import { REDIRECT_MAIN, NOT_FOUND_ROUTE } from './base';
import { MENU_GROUP_IDS } from '@/router/constants';
const modules = import.meta.glob('./modules/*.ts', { eager: true });
// const externalModules = import.meta.glob('./externalModules/*.ts', {

View File

@ -15,8 +15,8 @@ const COMPONENTS: AppRouteRecordRaw[] = [
locale: '全域数据引擎',
icon: IconBookmark,
requiresAuth: true,
requireLogin: true,
roles: ['*'],
requiresSidebar: true,
id: MENU_GROUP_IDS.DATA_ENGINE_ID,
},
children: [
@ -26,8 +26,8 @@ const COMPONENTS: AppRouteRecordRaw[] = [
meta: {
locale: '行业热门话题洞察',
requiresAuth: true,
requireLogin: true,
roles: ['*'],
menuId: 2,
},
component: () => import('@/views/components/dataEngine/hotTranslation.vue'),
},
@ -37,8 +37,8 @@ const COMPONENTS: AppRouteRecordRaw[] = [
meta: {
locale: '行业词云',
requiresAuth: true,
requireLogin: true,
roles: ['*'],
menuId: 3,
},
component: () => import('@/views/components/dataEngine/hotCloud.vue'),
},
@ -48,8 +48,8 @@ const COMPONENTS: AppRouteRecordRaw[] = [
meta: {
locale: '行业关键词动向',
requiresAuth: true,
requireLogin: true,
roles: ['*'],
menuId: 4,
},
component: () => import('@/views/components/dataEngine/keyWord.vue'),
},
@ -59,8 +59,8 @@ const COMPONENTS: AppRouteRecordRaw[] = [
meta: {
locale: '用户痛点观察',
requiresAuth: true,
requireLogin: true,
roles: ['*'],
menuId: 5,
},
component: () => import('@/views/components/dataEngine/userPainPoints.vue'),
},
@ -70,8 +70,8 @@ const COMPONENTS: AppRouteRecordRaw[] = [
meta: {
locale: '重点品牌动向',
requiresAuth: true,
requireLogin: true,
roles: ['*'],
menuId: 6,
},
component: () => import('@/views/components/dataEngine/keyBrandMovement.vue'),
},
@ -81,8 +81,8 @@ const COMPONENTS: AppRouteRecordRaw[] = [
meta: {
locale: '用户画像',
requiresAuth: true,
requireLogin: true,
roles: ['*'],
menuId: 7,
},
component: () => import('@/views/components/dataEngine/userPersona.vue'),
},

View File

@ -14,9 +14,9 @@ const COMPONENTS: AppRouteRecordRaw[] = [
meta: {
locale: '管理中心',
icon: IconBookmark,
requiresAuth: true,
requiresAuth: false,
requireLogin: true,
roles: ['*'],
requiresSidebar: true,
id: MENU_GROUP_IDS.MANAGEMENT_ID,
},
children: [
@ -26,7 +26,8 @@ const COMPONENTS: AppRouteRecordRaw[] = [
component: () => import('@/views/components/management/person'),
meta: {
locale: '个人信息',
requiresAuth: true,
requiresAuth: false,
requireLogin: true,
roles: ['*'],
},
},
@ -36,7 +37,8 @@ const COMPONENTS: AppRouteRecordRaw[] = [
component: () => import('@/views/components/management/enterprise'),
meta: {
locale: '企业信息',
requiresAuth: true,
requiresAuth: false,
requireLogin: true,
roles: ['*'],
},
},
@ -46,7 +48,8 @@ const COMPONENTS: AppRouteRecordRaw[] = [
component: () => import('@/views/components/management/account'),
meta: {
locale: '账号管理',
requiresAuth: true,
requiresAuth: false,
requireLogin: true,
roles: ['*'],
},
},

View File

@ -19,8 +19,8 @@ const COMPONENTS: AppRouteRecordRaw[] = [
locale: '品牌资产管理',
icon: IconRepository,
requiresAuth: true,
requireLogin: true,
roles: ['*'],
requiresSidebar: true,
id: MENU_GROUP_IDS.PROPERTY_ID,
},
children: [
@ -30,8 +30,8 @@ const COMPONENTS: AppRouteRecordRaw[] = [
meta: {
locale: '品牌信息',
requiresAuth: true,
requireLogin: true,
roles: ['*'],
menuId: 11,
},
component: () => import('@/views/property-marketing/brands/brand-materials/index.vue'),
},
@ -45,8 +45,8 @@ const COMPONENTS: AppRouteRecordRaw[] = [
locale: '账号资源中心',
icon: IconMediaAccount,
requiresAuth: true,
requireLogin: true,
roles: ['*'],
requiresSidebar: true,
id: MENU_GROUP_IDS.PROPERTY_ID,
},
children: [
@ -56,8 +56,8 @@ const COMPONENTS: AppRouteRecordRaw[] = [
meta: {
locale: '账号管理',
requiresAuth: true,
requireLogin: true,
roles: ['*'],
menuId: 12,
},
component: () => import('@/views/property-marketing/media-account/account-manage'),
},
@ -67,6 +67,7 @@ const COMPONENTS: AppRouteRecordRaw[] = [
meta: {
locale: '账号数据看板',
requiresAuth: true,
requireLogin: true,
roles: ['*'],
},
component: () => import('@/views/property-marketing/media-account/account-dashboard'),
@ -77,6 +78,7 @@ const COMPONENTS: AppRouteRecordRaw[] = [
meta: {
locale: '账号详情',
requiresAuth: true,
requireLogin: true,
roles: ['*'],
hideInMenu: true,
activeMenu: 'MediaAccountAccountDashboard',
@ -93,8 +95,8 @@ const COMPONENTS: AppRouteRecordRaw[] = [
locale: '投放资源中心',
icon: IconPutAccount,
requiresAuth: true,
requireLogin: true,
roles: ['*'],
requiresSidebar: true,
id: MENU_GROUP_IDS.PROPERTY_ID,
},
children: [
@ -104,8 +106,8 @@ const COMPONENTS: AppRouteRecordRaw[] = [
meta: {
locale: '账户管理',
requiresAuth: true,
requireLogin: true,
roles: ['*'],
menuId: 13,
},
component: () => import('@/views/property-marketing/put-account/account-manage'),
},
@ -115,6 +117,7 @@ const COMPONENTS: AppRouteRecordRaw[] = [
meta: {
locale: '账户数据',
requiresAuth: true,
requireLogin: true,
roles: ['*'],
},
component: () => import('@/views/property-marketing/put-account/account-data'),
@ -125,6 +128,7 @@ const COMPONENTS: AppRouteRecordRaw[] = [
meta: {
locale: '投放表现分析',
requiresAuth: true,
requireLogin: true,
roles: ['*'],
},
component: () => import('@/views/property-marketing/put-account/account-dashboard'),
@ -135,6 +139,7 @@ const COMPONENTS: AppRouteRecordRaw[] = [
meta: {
locale: '投放指南',
requiresAuth: true,
requireLogin: true,
roles: ['*'],
},
component: () => import('@/views/property-marketing/put-account/investment-guidelines'),
@ -161,8 +166,8 @@ const COMPONENTS: AppRouteRecordRaw[] = [
locale: '智能方案管理',
icon: IconIntelligentSolution,
requiresAuth: true,
requireLogin: true,
roles: ['*'],
requiresSidebar: true,
id: MENU_GROUP_IDS.PROPERTY_ID,
},
children: [
@ -172,8 +177,8 @@ const COMPONENTS: AppRouteRecordRaw[] = [
meta: {
locale: '业务洞察报告',
requiresAuth: true,
requireLogin: true,
roles: ['*'],
menuId: 14,
},
component: () => import('@/views/property-marketing/intelligent-solution/businessAnalysisReport'),
},
@ -183,6 +188,7 @@ const COMPONENTS: AppRouteRecordRaw[] = [
meta: {
locale: '竟品对比报告',
requiresAuth: true,
requireLogin: true,
roles: ['*'],
},
component: () => import('@/views/property-marketing/intelligent-solution/competitiveProductAnalysisReport'),

View File

@ -14,5 +14,6 @@ declare module 'vue-router' {
noAffix?: boolean; // if set true, the tag will not affix in the tab-bar
ignoreCache?: boolean; // if set true, the page will not be cached
hideSidebar?: boolean;
requireLogin?: boolean; // 是否需要登陆才能访问
}
}

View File

@ -1,4 +1,7 @@
import { fetchEnterpriseInfo } from '@/api/all/login';
import { useSidebarStore } from '@/stores/modules/side-bar';
import { useUserStore } from '@/stores/modules/user';
import { glsWithCatch, slsWithCatch, rlsWithCatch } from '@/utils/stroage';
interface EnterpriseInfo {
id: number;
@ -7,6 +10,7 @@ interface EnterpriseInfo {
used_update_name_count: number;
sub_account_quota: number;
used_sub_account_count: number;
permissions: string[];
}
interface EnterpriseState {
@ -15,24 +19,14 @@ interface EnterpriseState {
export const useEnterpriseStore = defineStore('enterprise', {
state: (): EnterpriseState => ({
enterpriseInfo: (() => {
const stored = localStorage.getItem('enterpriseInfo');
if (stored) {
try {
return JSON.parse(stored) as EnterpriseInfo;
} catch {
return null;
}
}
return null;
})(),
enterpriseInfo: (glsWithCatch('enterpriseInfo') && JSON.parse(glsWithCatch('enterpriseInfo') as string)) || null,
}),
actions: {
setEnterpriseInfo(enterpriseInfo: EnterpriseInfo) {
this.enterpriseInfo = enterpriseInfo;
localStorage.setItem('enterpriseInfo', JSON.stringify(enterpriseInfo));
slsWithCatch('enterpriseInfo', JSON.stringify(enterpriseInfo));
},
clearEnterpriseInfo() {
clearUserEnterpriseInfo() {
this.enterpriseInfo = null;
localStorage.removeItem('enterpriseInfo');
},
@ -51,14 +45,12 @@ export const useEnterpriseStore = defineStore('enterprise', {
this.enterpriseInfo.used_sub_account_count++;
}
},
getEnterpriseInfo(): EnterpriseInfo | null {
return this.enterpriseInfo;
},
async updateEnterpriseInfo() {
const res = await fetchEnterpriseInfo(this.enterpriseInfo!.id);
const { code, data } = res;
if (code === 200) {
this.setEnterpriseInfo(data);
async getEnterpriseInfo() {
if (this.enterpriseInfo) {
const { code, data } = await fetchEnterpriseInfo(this.enterpriseInfo!.id);
if (code === 200) {
this.setEnterpriseInfo(data);
}
}
},
},

View File

@ -0,0 +1,90 @@
import { MENU_GROUP_IDS } from '@/router/constants';
export const MENU_LIST = [
{
id: MENU_GROUP_IDS.WORK_BENCH_ID,
name: '工作台',
routeName: 'Home',
includeRouteNames: ['Home'],
requiresAuth: false,
permissionKey: '', // 权限key如果为空则表示该菜单不需要权限与后端约定
},
{
id: MENU_GROUP_IDS.DATA_ENGINE_ID,
name: '全域数据分析',
permissionKey: 'data_analysis',
requiresAuth: true,
children: [
{
name: '行业热门话题洞察',
routeName: 'DataEngineHotTranslation',
includeRouteNames: ['DataEngineHotTranslation'],
},
{
name: '行业词云',
routeName: 'DataEngineHotCloud',
includeRouteNames: ['DataEngineHotCloud'],
},
{
name: '行业关键词动向',
routeName: 'DataEngineKeyWord',
includeRouteNames: ['DataEngineKeyWord'],
},
{
name: '用户痛点观察',
routeName: 'DataEngineUserPainPoints',
includeRouteNames: ['DataEngineUserPainPoints'],
},
{
name: '重点品牌动向',
routeName: 'DataEngineKeyBrandMovement',
includeRouteNames: ['DataEngineKeyBrandMovement'],
},
{
name: '用户画像',
routeName: 'DataEngineUserPersona',
includeRouteNames: ['DataEngineUserPersona'],
},
],
},
{
id: MENU_GROUP_IDS.PROPERTY_ID,
name: '营销资产中台',
permissionKey: 'marketing_asset',
requiresAuth: true,
children: [
{
name: '品牌资产管理',
routeName: 'RepositoryBrandMaterials',
includeRouteNames: ['RepositoryBrandMaterials'],
},
{
name: '账号资源中心',
routeName: 'MediaAccountAccountManagement',
includeRouteNames: [
'MediaAccountAccountManagement',
'MediaAccountAccountDashboard',
'MediaAccountAccountDetails',
],
},
{
name: '投放资源中心',
routeName: 'PutAccountAccountManagement',
includeRouteNames: [
'PutAccountAccountManagement',
'PutAccountAccountData',
'PutAccountAccountDashboard',
'PutAccountInvestmentGuidelines',
'guideDetail',
],
},
{
name: '智能方案管理',
routeName: 'IntelligentSolutionBusinessAnalysisReport',
includeRouteNames: [
'IntelligentSolutionBusinessAnalysisReport',
'IntelligentSolutionCompetitiveProductAnalysisReport',
],
},
],
},
];

View File

@ -3,19 +3,23 @@
* @Date: 2025-06-23 22:13:30
*/
import { defineStore } from 'pinia';
import { MENU_GROUP_IDS } from '@/router/constants';
import { appRoutes } from '@/router/routes';
import router from '@/router';
import type { RouteLocationNormalized } from 'vue-router';
const { DATA_ENGINE_ID, MANAGEMENT_ID } = MENU_GROUP_IDS;
import { MENU_LIST } from './constants';
import { useEnterpriseStore } from '@/stores/modules/enterprise';
interface sidebarState {
activeMenuId: number | null;
menuList: any[];
allowAccessRoutes: any[];
}
export const useSidebarStore = defineStore('sidebar', {
state: (): sidebarState => ({
activeMenuId: null,
menuList: [],
allowAccessRoutes: [], // 允许访问的路由列表
}),
actions: {
clearActiveMenuId() {
@ -24,14 +28,25 @@ export const useSidebarStore = defineStore('sidebar', {
setActiveMenuId(id: number) {
this.activeMenuId = id;
},
clearUserNavbarMenuList() {
this.menuList = [];
},
// navbar菜单列表由企业对应权限决定
getUserNavbarMenuList() {
const enterpriseStore = useEnterpriseStore();
this.menuList = MENU_LIST.filter(
(item) => !item.permissionKey || enterpriseStore.enterpriseInfo?.permissions?.includes(item.permissionKey),
);
},
// 根据当前路由自动设置 activeMenuId
setActiveMenuIdByRoute(route: RouteLocationNormalized) {
// console.log('setActiveMenuIdByRoute ');
const appRoutes = router.options?.routes ?? [];
// 查找当前路由所属的菜单组
const findMenuGroup = (routes: any[]): number | null => {
for (const routeItem of routes) {
// 检查子路由
if (routeItem.children && routeItem.children.length > 0) {
if (routeItem.children?.length > 0) {
// 检查当前路由是否是这个父路由的子路由
const isChildRoute = routeItem.children.some((child: any) => child.name === route.name);
if (isChildRoute) {
@ -42,12 +57,16 @@ export const useSidebarStore = defineStore('sidebar', {
if (childResult !== null) {
return routeItem.meta?.id || childResult;
}
} else {
if (routeItem.name === route.name) {
return routeItem.meta?.id || null;
}
}
}
return null;
};
const menuId = findMenuGroup(appRoutes);
const menuId = findMenuGroup(appRoutes as any);
if (menuId !== null) {
this.activeMenuId = menuId;
}

View File

@ -1,5 +1,10 @@
import type { RouteRecordNormalized } from 'vue-router';
import { defineStore } from 'pinia';
import { fetchProfileInfo } from '@/api/all/login';
import { useSidebarStore } from '@/stores/modules/side-bar';
import router from '@/router';
import { glsWithCatch, slsWithCatch, rlsWithCatch } from '@/utils/stroage';
interface UserInfo {
id: number;
@ -10,16 +15,10 @@ interface UserInfo {
// 添加其他用户属性...
}
interface CompanyInfo {
id: number;
name: string;
// 添加其他公司属性...
}
interface UserState {
token: string;
userInfo: UserInfo | null;
companyInfo: CompanyInfo | null;
allowAccessRoutes: string[];
// isLogin: boolean;
}
@ -31,9 +30,10 @@ interface UserInfo {
export const useUserStore = defineStore('user', {
state: (): UserState => ({
token: localStorage.getItem('accessToken') || '',
userInfo: null,
companyInfo: null,
token: glsWithCatch('accessToken') || '',
userInfo: (glsWithCatch('userInfo') && JSON.parse(glsWithCatch('userInfo') as string)) || null,
allowAccessRoutes:
(glsWithCatch('allowAccessRoutes') && JSON.parse(glsWithCatch('allowAccessRoutes') as string)) || [], // 允许访问的路由列表
}),
getters: {
isLogin(): boolean {
@ -44,12 +44,12 @@ export const useUserStore = defineStore('user', {
// 设置 Token
setToken(token: string) {
this.token = `Bearer ${token}`;
localStorage.setItem('accessToken', this.token);
slsWithCatch('accessToken', this.token);
},
deleteToken() {
this.token = '';
localStorage.removeItem('accessToken');
rlsWithCatch('accessToken');
},
// 获取 Token
@ -60,14 +60,54 @@ export const useUserStore = defineStore('user', {
// 设置用户信息
setUserInfo(userInfo: UserInfo | null) {
this.userInfo = userInfo;
slsWithCatch('userInfo', JSON.stringify(userInfo));
},
clearUserInfo() {
this.userInfo = null;
rlsWithCatch('userInfo');
},
// 获取用户信息
async fetchUserInfo() {
async getUserInfo() {
const { code, data } = await fetchProfileInfo();
if (code === 200) {
this.setUserInfo(data);
}
},
clearUserAllowAccessRoutes() {
this.allowAccessRoutes = [];
rlsWithCatch('allowAccessRoutes');
},
getUserAllowAccessRoutes() {
const sidebarStore = useSidebarStore();
const menuList = sidebarStore.menuList;
const appRoutes = router.getRoutes();
appRoutes.forEach((route: any) => {
if (!route.meta?.requiresAuth) {
this.allowAccessRoutes.push(route.name);
}
});
const pushAllowAccessRoutes = (includeRouteNames: string[]) => {
const matchedRoute = appRoutes
.filter((route: any) => includeRouteNames.includes(route.name))
.map((route: any) => route.name);
this.allowAccessRoutes.push(...matchedRoute);
};
menuList.forEach((item) => {
if (item.children && item.children.length > 0) {
item.children.forEach((child: any) => {
pushAllowAccessRoutes(child.includeRouteNames);
});
} else {
pushAllowAccessRoutes(item.includeRouteNames);
}
});
this.allowAccessRoutes = uniq(this.allowAccessRoutes);
slsWithCatch('allowAccessRoutes', JSON.stringify(this.allowAccessRoutes));
},
},
});

View File

@ -1,24 +1,30 @@
@font-face {
font-family: 'PuHuiTi-Medium';
src: url('@/assets/fonts/Alibaba-PuHuiTi-Medium.otf');
font-family: 'PuHuiTi-Regular';
src: url('@/assets/fonts/Alibaba-PuHuiTi-Regular.woff2') format('woff2');
font-display: swap;
}
@font-face {
font-family: 'PuHuiTi-Regular';
src: url('@/assets/fonts/Alibaba-PuHuiTi-Regular.otf');
font-family: 'PuHuiTi-Medium';
src: url('@/assets/fonts/Alibaba-PuHuiTi-Medium.woff2') format('woff2');
font-display: swap;
}
@font-face {
font-family: 'PuHuiTi-Bold';
src: url('@/assets/fonts/Alibaba-PuHuiTi-Bold.otf');
src: url('@/assets/fonts/Alibaba-PuHuiTi-Bold.woff2') format('woff2');
font-display: swap;
}
// 使用统一的字体族名
.font-family-puhui-regular {
font-family: 'PuHuiTi-Regular' !important;
}
.font-family-puhui-medium {
font-family: 'PuHuiTi-Medium' !important;
}
.font-family-puhui-bold {
font-family: PuHuiTi-Bold !important;
}
.font-family-puhui-medium {
font-family: PuHuiTi-Medium !important;
}
.font-family-puhui-regular {
font-family: PuHuiTi-Regular !important;
font-family: 'PuHuiTi-Bold' !important;
}

47
src/utils/stroage.ts Normal file
View File

@ -0,0 +1,47 @@
export const glsWithCatch = (key: string) => {
try {
return localStorage?.getItem(key);
} catch (error) {
console.log(error);
}
};
export const slsWithCatch = (key: string, value: any) => {
try {
localStorage?.setItem(key, value);
} catch (error) {
console.log(error);
}
};
export const rlsWithCatch = (key: string) => {
try {
localStorage?.removeItem(key);
} catch (error) {
console.log(error);
}
};
export const gssWithCatch = (key: string) => {
try {
return sessionStorage?.getItem(key);
} catch (error) {
console.log(error);
}
};
export const sssWithCatch = (key: string, value: any) => {
try {
sessionStorage?.setItem(key, value);
} catch (error) {
console.log(error);
}
};
export const rssWithCatch = (key: string) => {
try {
sessionStorage?.removeItem(key);
} catch (error) {
console.log(error);
}
};

View File

@ -12,9 +12,24 @@ import { useSidebarStore } from '@/stores/modules/side-bar';
export function goUserLogin(query?: any) {
router.push({ name: 'UserLogin', query });
}
// 初始化企业信息、navbar菜单、允许访问的路由
export const getUserEnterpriseInfo = async () => {
const enterpriseStore = useEnterpriseStore();
const sidebarStore = useSidebarStore();
const userStore = useUserStore();
await enterpriseStore.getEnterpriseInfo();
sidebarStore.getUserNavbarMenuList(); // 初始化navbar菜单
userStore.getUserAllowAccessRoutes(); // 初始化允许访问的路由
};
// 登录处理
export async function handleUserLogin() {
const userStore = useUserStore();
await userStore.getUserInfo(); // 初始化用户信息
await getUserEnterpriseInfo();
handleUserHome();
}
@ -28,9 +43,13 @@ export function handleUserLogout() {
const enterpriseStore = useEnterpriseStore();
const sidebarStore = useSidebarStore();
userStore.deleteToken();
enterpriseStore.clearEnterpriseInfo();
userStore.clearUserInfo();
enterpriseStore.clearUserEnterpriseInfo();
sidebarStore.clearUserNavbarMenuList();
userStore.clearUserAllowAccessRoutes();
sidebarStore.clearActiveMenuId();
userStore.deleteToken();
goUserLogin();
}

View File

@ -225,7 +225,6 @@ const clearError = (field: string) => {
const handleOk = async () => {
visible.value = false;
await enterpriseStore.updateEnterpriseInfo();
handleUserLogin();
};
@ -269,11 +268,9 @@ const getProfileInfo = async () => {
mobileNumber.value = data['mobile'];
accounts.value = enterprises;
enterpriseStore.setEnterpriseInfo(data);
userStore.setUserInfo(data);
if (enterprises.length > 0) {
if (enterprises.length === 1) {
await enterpriseStore.updateEnterpriseInfo();
handleUserLogin();
} else {
// 多个企业时候需要弹窗让用户选择企业

View File

@ -99,7 +99,7 @@ const addAccountVisible = ref(false);
const deleteVisible = ref(false);
const deleteTitle = ref('');
const enterpriseInfo = store.getEnterpriseInfo();
const enterpriseInfo = store.enterpriseInfo;
const okText = computed(() => {
if (!canAddAccount.value) {

View File

@ -48,7 +48,7 @@ const form = reactive({
name: '',
});
const enterpriseInfo = store.getEnterpriseInfo();
const enterpriseInfo = store.enterpriseInfo;
const columns = [
{

View File

@ -26,7 +26,7 @@
v-if="props.product.status === Status.Enable || props.product.status === Status.ON_TRIAL"
class="primary-button"
type="primary"
@click="gotoModule(props.product.menu_id)"
@click="gotoModule(props.product.id)"
>
进入模块
</a-button>
@ -70,13 +70,16 @@ import { trialProduct } from '@/api/all';
import { useRouter } from 'vue-router';
import CustomerServiceModal from '@/components/customer-service-modal.vue';
import { appRoutes } from '@/router/routes';
import { useSidebarStore } from '@/stores/modules/side-bar';
import { useEnterpriseStore } from '@/stores/modules/enterprise';
import { useUserStore } from '@/stores';
import { getUserEnterpriseInfo } from '@/utils/user';
const props = defineProps<{
product: Product;
}>();
const emit = defineEmits(['refresh']);
const sidebarStore = useSidebarStore();
enum Status {
Disable = 0, // 禁用
@ -87,30 +90,37 @@ enum Status {
}
interface Product {
id: number;
status: Status;
name: string;
image: string;
desc: string;
menu_id: number;
id: number;
expired_at?: number;
}
const visible = ref(false);
const router = useRouter();
const enterpriseStore = useEnterpriseStore();
const userStore = useUserStore();
const sidebarStore = useSidebarStore();
const handleTrial = async (id: any) => {
await trialProduct(id);
AMessage.success('试用成功!');
emit('refresh');
const { code } = await trialProduct(id);
if (code === 200) {
getUserEnterpriseInfo();
AMessage.success('试用成功!');
emit('refresh');
}
};
const gotoModule = (menuId: number) => {
const _target = appRoutes.find((v) => v.meta.id === menuId);
if (_target) {
console.log({ _target });
router.push({ name: _target.name });
}
const routeMap: Record<number, string> = {
'1': 'DataEngineHotTranslation',
'2': 'RepositoryBrandMaterials',
};
console.log(routeMap[menuId]);
router.push({ name: routeMap[menuId] });
};
</script>
<style scoped lang="less">

View File

@ -87,15 +87,12 @@
</a-form-item>
</template>
<a-form-item label="手机号" field="mobile" required>
<a-form-item label="手机号" field="mobile" required>
<a-input v-model="form.mobile" placeholder="请输入..." size="large" />
</a-form-item>
<a-form-item label="运营人员" field="operator_name" required>
<a-input v-model="form.operator_name" placeholder="请输入..." class="w-240px" size="large" />
</a-form-item>
<a-form-item label="号码持有人" field="holder_name" required>
<a-input v-model="form.holder_name" placeholder="请输入..." class="w-240px" size="large" />
</a-form-item>
<a-form-item label="运营平台" :required="!isEdit">
<img v-if="isEdit" :src="form.platform === 0 ? icon3 : icon4" width="24" height="24" />
<a-radio-group v-else v-model="form.platform">
@ -103,6 +100,9 @@
<a-radio :value="1">小红书</a-radio>
</a-radio-group>
</a-form-item>
<a-form-item label="号码持有人" field="holder_name">
<a-input v-model="form.holder_name" placeholder="请输入..." class="w-240px" size="large" />
</a-form-item>
<a-form-item label="选择分组">
<GroupSelect
v-model="form.group_id"
@ -130,6 +130,26 @@
:auto-size="{ minRows: 3, maxRows: 5 }"
/>
</a-form-item>
<template v-if="!isEdit">
<a-form-item label="Cookie值">
<template #label>
<span class="label">Cookie值</span>
<a-tooltip content="开启后可直接填写 Cookie无需扫码授权">
<icon-question-circle size="14" class="ml-4px color-#737478" />
</a-tooltip>
</template>
<a-switch v-model="isCustomCookie" size="large" />
</a-form-item>
<a-form-item v-if="isCustomCookie" label="" field="cookie">
<a-textarea
v-model="form.cookie"
placeholder="请输入..."
size="large"
max-length="72"
:auto-size="{ minRows: 3, maxRows: 5 }"
/>
</a-form-item>
</template>
</template>
</a-form>
<template #footer>
@ -181,7 +201,8 @@ const INITIAL_FORM = {
platform: 0,
group_id: undefined,
tag_ids: [],
end_work_link: '',
end_work_link: undefined,
cookie: undefined,
};
const groupOptions = ref([]);
@ -197,6 +218,7 @@ const file = ref(null);
const authorizedAccountModalRef = ref(null);
const importPromptModalRef = ref(null);
const uploadRef = ref(null);
const isCustomCookie = ref(false);
const form = ref(cloneDeep(INITIAL_FORM));
const rules = {
@ -218,7 +240,7 @@ const rules = {
},
],
operator_name: [{ required: true, message: '请输入运营人员' }],
holder_name: [{ required: true, message: '请输入号码持有人' }],
// holder_name: [{ required: true, message: '请输入号码持有人' }],
};
const isBatchImport = computed(() => uploadType.value === 'batch');
@ -266,6 +288,7 @@ const reset = () => {
fileName.value = '';
file.value = null;
isEdit.value = false;
isCustomCookie.value = false;
uploadStatus.value = UploadStatus.DEFAULT;
uploadType.value = 'manual';
};
@ -320,6 +343,30 @@ const handleBatchImport = async () => {
}
};
const handleAddAccount = async () => {
const _isCustomCookie = isCustomCookie.value;
const { code, data } = await postMediaAccounts({
...form.value,
cookie: _isCustomCookie ? form.value.cookie : undefined,
});
if (code === 200) {
emits('update');
onClose();
const { id, platform } = data;
!_isCustomCookie && startAuthorized(id, platform);
}
};
const handleEditAccount = async () => {
const { code } = await putMediaAccounts({ id: id.value, ...form.value });
if (code === 200) {
AMessage.success('修改成功');
emits('update');
onClose();
}
};
async function onSubmit() {
if (isBatchImport.value) {
handleBatchImport();
@ -328,17 +375,7 @@ async function onSubmit() {
formRef.value.validate(async (errors) => {
if (!errors) {
const _fn = id.value ? putMediaAccounts : postMediaAccounts;
const _params = id.value ? { id: id.value, ...form.value } : form.value;
const { code, data } = await _fn(_params);
if (code === 200) {
isEdit.value && AMessage.success('修改成功');
emits('update');
onClose();
if (!isEdit.value) {
startAuthorized(data?.id, data?.platform);
}
}
isEdit.value ? handleEditAccount() : handleAddAccount();
}
});
}

View File

@ -34,7 +34,7 @@
<div class="img-box">
<template v-if="qrCodeLoading || isFailLoadQrCode">
<div class="relative w-160px h-160px">
<a-image v-if="qrCodeLoading" :src="icon1" width="160" height="160" />
<a-image :src="icon1" width="160" height="160" />
<div class="absolute top-0 left-0 z-2 w-full h-full flex flex-col items-center justify-center">
<img v-if="isFailLoadQrCode" :src="icon4" width="24" height="24" class="mb-13px" />
<icon-loading v-else size="24" class="color-#6D4CFE mb-13px" />
@ -49,7 +49,7 @@
</template>
<a-image v-else :src="qrCodeUrl" width="160" height="160" />
<div v-if="isOverdue" class="mask cursor-pointer" @click="getAuthorizedQrCode">
<div v-if="isOverdue" class="mask cursor-pointer" @click="handleRefreshQrCode">
<icon-refresh size="24" class="mb-13px" />
<p class="s1">二维码失效</p>
<p class="s1">请点击刷新</p>
@ -225,11 +225,15 @@ const clearOverdueTimer = () => {
overdueTimer = null;
}
};
// 重新扫码
const handleRefreshQrCode = () => {
resetTaskFields();
getAuthorizedQrCode();
};
const handleOk = () => {
if (isFailLoadQrCode.value) {
resetTaskFields();
getAuthorizedQrCode();
handleRefreshQrCode();
return;
}
@ -249,8 +253,7 @@ const handleOk = () => {
if (isSuccess.value) {
close();
} else {
resetTaskFields();
getAuthorizedQrCode();
handleRefreshQrCode();
}
} else {
startLoading();

View File

@ -34,7 +34,7 @@
<div class="img-box">
<template v-if="qrCodeLoading || isFailLoadQrCode">
<div class="relative w-160px h-160px">
<a-image v-if="qrCodeLoading" :src="icon1" width="160" height="160" />
<a-image :src="icon1" width="160" height="160" />
<div class="absolute top-0 left-0 z-2 w-full h-full flex flex-col items-center justify-center">
<img v-if="isFailLoadQrCode" :src="icon4" width="24" height="24" class="mb-13px" />
<icon-loading v-else size="24" class="color-#6D4CFE mb-13px" />
@ -49,7 +49,7 @@
</template>
<a-image v-else :src="qrCodeUrl" width="160" height="160" />
<div v-if="isOverdue" class="mask cursor-pointer" @click="getAuthorizedQrCode">
<div v-if="isOverdue" class="mask cursor-pointer" @click="handleRefreshQrCode">
<icon-refresh size="24" class="mb-13px" />
<p class="s1">二维码失效</p>
<p class="s1">请点击刷新</p>
@ -277,11 +277,16 @@ const clearStatusPollingTimer = () => {
}
};
// 重新扫码
const handleRefreshQrCode = () => {
resetTaskFields();
getAuthorizedQrCode();
};
const handleOk = () => {
if (taskStep.value === TASK_STEP.default) {
if (isFailLoadQrCode.value) {
resetTaskFields();
getAuthorizedQrCode();
handleRefreshQrCode();
return;
}
@ -308,8 +313,7 @@ const handleOk = () => {
if (isSuccess.value) {
close();
} else {
resetTaskFields();
getAuthorizedQrCode();
handleRefreshQrCode();
}
}
}