2025-04-30 01:43:23 +08:00
|
|
|
|
<template>
|
|
|
|
|
|
<view class="home-page">
|
|
|
|
|
|
<view class="content-container">
|
|
|
|
|
|
<!-- 用户信息卡片 -->
|
2025-06-23 18:20:19 +08:00
|
|
|
|
<view class="user-info-card" v-if="curXs">
|
2025-09-15 23:28:07 +08:00
|
|
|
|
<view class="banner-content">
|
|
|
|
|
|
<image src="/static/base/home/2211.png" class="banner-img"></image>
|
|
|
|
|
|
<view class="banner-overlay">
|
|
|
|
|
|
</view>
|
|
|
|
|
|
</view>
|
2025-06-29 19:24:41 +08:00
|
|
|
|
<!-- 学生信息 -->
|
|
|
|
|
|
<XsPicker :is-bar="false" @change="switchXs" />
|
2025-09-15 23:28:07 +08:00
|
|
|
|
|
2025-06-29 19:24:41 +08:00
|
|
|
|
<view class="glxs-btn" @click="goToGlxs">
|
|
|
|
|
|
<u-icon name="plus" size="12" color="#fff"></u-icon>
|
|
|
|
|
|
<text>新增学生</text>
|
2025-04-30 01:43:23 +08:00
|
|
|
|
</view>
|
|
|
|
|
|
</view>
|
|
|
|
|
|
|
|
|
|
|
|
<!-- 功能菜单 -->
|
2025-05-30 17:22:30 +08:00
|
|
|
|
<view class="menu-section">
|
|
|
|
|
|
<view class="section-title">
|
|
|
|
|
|
<text class="title-text">校园服务</text>
|
|
|
|
|
|
<view class="title-line"></view>
|
|
|
|
|
|
</view>
|
|
|
|
|
|
<view class="grid-menu">
|
2026-02-23 17:31:01 +08:00
|
|
|
|
<view v-for="(item, index) in menuItems" :key="item.id ?? index"
|
2025-09-15 23:28:07 +08:00
|
|
|
|
class="grid-item" @click="handleMenuClick(item)">
|
2025-05-30 17:22:30 +08:00
|
|
|
|
<view class="grid-icon-container">
|
|
|
|
|
|
<view class="icon-background"></view>
|
2026-02-04 09:20:42 +08:00
|
|
|
|
<image :src="item.icon" class="grid-icon" @error="handleImageError" mode="aspectFit"></image>
|
2025-05-30 17:22:30 +08:00
|
|
|
|
</view>
|
|
|
|
|
|
<text class="grid-text">{{ item.title }}</text>
|
2025-05-06 13:03:49 +08:00
|
|
|
|
</view>
|
2025-04-30 01:43:23 +08:00
|
|
|
|
</view>
|
|
|
|
|
|
</view>
|
|
|
|
|
|
|
|
|
|
|
|
<!-- 通知公告 -->
|
|
|
|
|
|
<view class="notice-section">
|
2025-05-30 17:22:30 +08:00
|
|
|
|
<view class="section-title">
|
|
|
|
|
|
<text class="title-text">通知公告</text>
|
|
|
|
|
|
<view class="title-line"></view>
|
2025-04-30 01:43:23 +08:00
|
|
|
|
</view>
|
|
|
|
|
|
|
|
|
|
|
|
<view class="notice-list">
|
2025-09-15 23:28:07 +08:00
|
|
|
|
<view v-for="(notice, index) in announcements" :key="index" class="notice-item" @click="goToDetail(notice)">
|
2025-05-30 17:22:30 +08:00
|
|
|
|
<view class="notice-icon">
|
|
|
|
|
|
<u-icon name="bell" size="20" color="#4A90E2"></u-icon>
|
|
|
|
|
|
</view>
|
2025-04-30 01:43:23 +08:00
|
|
|
|
<view class="notice-content">
|
|
|
|
|
|
<text class="notice-title">{{ notice.title }}</text>
|
|
|
|
|
|
<text class="notice-desc">{{ notice.description }}</text>
|
2025-05-30 17:22:30 +08:00
|
|
|
|
<view class="notice-footer">
|
2025-06-15 20:18:13 +08:00
|
|
|
|
<text class="notice-date">{{ notice.releaseTime }}</text>
|
2025-05-30 17:22:30 +08:00
|
|
|
|
<view class="notice-arrow">
|
|
|
|
|
|
<u-icon name="arrow-right" size="12" color="#C8C9CC"></u-icon>
|
|
|
|
|
|
</view>
|
|
|
|
|
|
</view>
|
2025-04-30 01:43:23 +08:00
|
|
|
|
</view>
|
|
|
|
|
|
</view>
|
|
|
|
|
|
</view>
|
|
|
|
|
|
</view>
|
|
|
|
|
|
</view>
|
|
|
|
|
|
|
|
|
|
|
|
</view>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
|
|
|
|
|
|
<script setup lang="ts">
|
2026-02-23 17:31:01 +08:00
|
|
|
|
import { ref, computed, onMounted, watch, reactive } from "vue";
|
2025-06-29 19:24:41 +08:00
|
|
|
|
import XsPicker from "@/pages/base/components/XsPicker/index.vue"
|
2025-08-02 10:47:16 +08:00
|
|
|
|
import { getNoticeListApi } from "@/api/base/notice";
|
2026-02-23 17:31:01 +08:00
|
|
|
|
import { getMobileMenuApi } from "@/api/system/menu";
|
2025-06-14 22:28:04 +08:00
|
|
|
|
import { useUserStore } from "@/store/modules/user";
|
2025-06-15 20:18:13 +08:00
|
|
|
|
import { useDataStore } from "@/store/modules/data";
|
2026-02-23 17:31:01 +08:00
|
|
|
|
import { useMenuStore } from "@/store/modules/menu";
|
|
|
|
|
|
import { type MobileMenuTreeNode } from "@/api/system/menu";
|
2025-09-02 15:45:06 +08:00
|
|
|
|
import { PageUtils } from "@/utils/pageUtil";
|
2025-09-03 19:46:26 +08:00
|
|
|
|
import { useDebounce } from "@/utils/debounce";
|
2025-08-02 10:47:16 +08:00
|
|
|
|
|
2026-02-23 17:31:01 +08:00
|
|
|
|
const userStore = useUserStore();
|
|
|
|
|
|
const { getCurXs } = userStore;
|
|
|
|
|
|
const dataStore = useDataStore();
|
|
|
|
|
|
const menuStore = useMenuStore();
|
|
|
|
|
|
const { setData, getAppCode, setGlobal } = dataStore;
|
2025-09-03 19:46:26 +08:00
|
|
|
|
const { debounce } = useDebounce(2000);
|
2025-06-14 22:28:04 +08:00
|
|
|
|
|
2026-02-23 17:31:01 +08:00
|
|
|
|
/** 家长端菜单项(后端 getMobileMenuApi 动态加载) */
|
|
|
|
|
|
interface HomeMenuItem {
|
|
|
|
|
|
id?: number | string;
|
|
|
|
|
|
title: string;
|
|
|
|
|
|
icon: string;
|
|
|
|
|
|
path?: string;
|
|
|
|
|
|
permissionKey?: string;
|
|
|
|
|
|
action?: string;
|
|
|
|
|
|
lxId?: string;
|
|
|
|
|
|
}
|
2025-09-15 23:28:07 +08:00
|
|
|
|
|
2026-02-23 17:31:01 +08:00
|
|
|
|
/** 家长端特殊菜单:选课/退费/缴费需 lxId + action,后端 auth_code 映射 */
|
|
|
|
|
|
const JZD_SPECIAL_MENU: Record<string, { lxId: string; action: string }> = {
|
|
|
|
|
|
"school-xqkxk": { lxId: "962488654", action: "jf" }, // 兴趣课选课
|
|
|
|
|
|
"school-jlbxk": { lxId: "816059832", action: "jf" }, // 俱乐部选课
|
|
|
|
|
|
"school-jcjf": { lxId: "JC", action: "jf" }, // 就餐报名
|
|
|
|
|
|
"school-xqk-tf": { lxId: "962488654", action: "tf" }, // 兴趣课退费
|
|
|
|
|
|
"school-jlb-tf": { lxId: "816059832", action: "tf" }, // 俱乐部退费
|
2025-07-21 20:21:57 +08:00
|
|
|
|
};
|
|
|
|
|
|
|
2026-02-23 17:31:01 +08:00
|
|
|
|
// 将菜单树扁平化为家长端结构
|
|
|
|
|
|
function flattenMenuToItems(nodes: MobileMenuTreeNode[]): HomeMenuItem[] {
|
|
|
|
|
|
const items: HomeMenuItem[] = [];
|
|
|
|
|
|
const walk = (list: MobileMenuTreeNode[]) => {
|
|
|
|
|
|
for (const node of list || []) {
|
|
|
|
|
|
if (node.pagePath) {
|
|
|
|
|
|
const icon = (node.normalCss && /^[a-zA-Z0-9_-]+$/.test(node.normalCss))
|
|
|
|
|
|
? `/static/base/home/${node.normalCss}.png`
|
|
|
|
|
|
: "/static/base/home/file-text-line.png";
|
|
|
|
|
|
const authCode = node.authCode || "";
|
|
|
|
|
|
const special = authCode ? JZD_SPECIAL_MENU[authCode] : undefined;
|
|
|
|
|
|
items.push({
|
|
|
|
|
|
id: node.id,
|
|
|
|
|
|
title: node.screenName,
|
|
|
|
|
|
icon,
|
|
|
|
|
|
path: node.pagePath,
|
|
|
|
|
|
permissionKey: authCode,
|
|
|
|
|
|
...(special && { lxId: special.lxId, action: special.action }),
|
|
|
|
|
|
});
|
2025-07-21 20:21:57 +08:00
|
|
|
|
}
|
2026-02-23 17:31:01 +08:00
|
|
|
|
if (node.children?.length) walk(node.children);
|
2025-07-21 20:21:57 +08:00
|
|
|
|
}
|
2026-02-23 17:31:01 +08:00
|
|
|
|
};
|
|
|
|
|
|
if (nodes.length === 1 && !nodes[0].pagePath && (nodes[0].children?.length ?? 0) > 0) {
|
|
|
|
|
|
walk(nodes[0].children!);
|
|
|
|
|
|
} else {
|
|
|
|
|
|
walk(nodes);
|
2025-07-21 20:21:57 +08:00
|
|
|
|
}
|
2026-02-23 17:31:01 +08:00
|
|
|
|
return items;
|
|
|
|
|
|
}
|
2025-07-21 20:21:57 +08:00
|
|
|
|
|
2026-02-23 17:31:01 +08:00
|
|
|
|
const menuItems = ref<HomeMenuItem[]>([]);
|
2025-04-30 01:43:23 +08:00
|
|
|
|
|
|
|
|
|
|
// 通知公告数据
|
2025-09-01 22:34:40 +08:00
|
|
|
|
const announcements = ref<any>([])
|
2025-04-30 01:43:23 +08:00
|
|
|
|
|
|
|
|
|
|
// 当前选中的学生
|
2025-06-29 19:24:41 +08:00
|
|
|
|
let curXs = computed(() => getCurXs)
|
2025-04-30 01:43:23 +08:00
|
|
|
|
|
2025-06-15 20:18:13 +08:00
|
|
|
|
let pageParams = ref({
|
2025-09-15 23:28:07 +08:00
|
|
|
|
page: 1,
|
|
|
|
|
|
rows: 10,
|
2025-06-23 18:20:19 +08:00
|
|
|
|
appCode: getAppCode
|
2025-06-15 20:18:13 +08:00
|
|
|
|
})
|
|
|
|
|
|
|
2025-06-24 20:33:42 +08:00
|
|
|
|
const goToGlxs = () => {
|
2025-09-15 23:28:07 +08:00
|
|
|
|
uni.navigateTo({
|
|
|
|
|
|
url: "/pages/base/home/glxs",
|
|
|
|
|
|
});
|
2025-06-24 20:33:42 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-04-30 01:43:23 +08:00
|
|
|
|
// 处理菜单点击
|
2025-09-03 19:46:26 +08:00
|
|
|
|
const handleMenuClick = debounce(async (item: any) => {
|
2025-04-30 01:43:23 +08:00
|
|
|
|
if (item.path) {
|
2025-09-15 21:06:37 +08:00
|
|
|
|
if (item.lxId) {
|
|
|
|
|
|
setGlobal({ lxId: item.lxId, action: item.action, from: 'home' });
|
|
|
|
|
|
PageUtils.toHome(item.lxId, item.action);
|
|
|
|
|
|
} else {
|
2025-08-11 10:30:23 +08:00
|
|
|
|
uni.navigateTo({
|
|
|
|
|
|
url: item.path,
|
|
|
|
|
|
});
|
|
|
|
|
|
}
|
2025-04-30 01:43:23 +08:00
|
|
|
|
}
|
2025-09-03 19:46:26 +08:00
|
|
|
|
});
|
2025-04-30 01:43:23 +08:00
|
|
|
|
|
|
|
|
|
|
// 切换学生
|
2025-06-23 18:20:19 +08:00
|
|
|
|
function switchXs(xs: any) {
|
|
|
|
|
|
getArticleList();
|
2025-04-30 01:43:23 +08:00
|
|
|
|
}
|
2025-06-14 22:28:04 +08:00
|
|
|
|
|
2026-02-04 09:20:42 +08:00
|
|
|
|
// 处理图片加载失败
|
|
|
|
|
|
function handleImageError(e: any) {
|
|
|
|
|
|
// 图片加载失败时,使用默认占位图标
|
|
|
|
|
|
const target = e.target || e.currentTarget;
|
|
|
|
|
|
if (target) {
|
|
|
|
|
|
target.src = '/static/base/home/file-text-line.png'; // 默认图标
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-06-15 20:18:13 +08:00
|
|
|
|
// 跳转到详情页面
|
|
|
|
|
|
function goToDetail(notice: any) {
|
2025-09-15 23:28:07 +08:00
|
|
|
|
setData(notice)
|
|
|
|
|
|
uni.navigateTo({
|
|
|
|
|
|
url: '/pages/base/home/detail'
|
|
|
|
|
|
})
|
2025-06-15 20:18:13 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-09-15 23:28:07 +08:00
|
|
|
|
const getArticleList = async () => {
|
2025-07-21 20:21:57 +08:00
|
|
|
|
if (curXs.value && curXs.value.njmcId) {
|
2025-08-02 10:47:16 +08:00
|
|
|
|
const params = {
|
|
|
|
|
|
page: pageParams.value.page,
|
|
|
|
|
|
rows: pageParams.value.rows,
|
|
|
|
|
|
appCode: getAppCode,
|
|
|
|
|
|
fbfw: 'JZ', // 发布范围:家长端
|
|
|
|
|
|
xqId: curXs.value.xqId || curXs.value.xq_id, // 学生所在学期
|
|
|
|
|
|
fbNjmcId: curXs.value.njmcId, // 学生所在年级
|
|
|
|
|
|
releaseFlag: 'A' // 发布状态:有效
|
|
|
|
|
|
};
|
2025-09-15 23:28:07 +08:00
|
|
|
|
|
2025-08-02 10:47:16 +08:00
|
|
|
|
getNoticeListApi(params).then(res => {
|
|
|
|
|
|
announcements.value = res.rows;
|
|
|
|
|
|
})
|
2025-09-15 23:28:07 +08:00
|
|
|
|
.catch((error) => {
|
|
|
|
|
|
// 接口调用失败
|
|
|
|
|
|
});
|
2025-07-21 20:21:57 +08:00
|
|
|
|
}
|
2025-06-23 18:20:19 +08:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
onMounted(async () => {
|
2025-09-03 15:02:20 +08:00
|
|
|
|
setGlobal({ from: 'home' });
|
2026-02-23 17:31:01 +08:00
|
|
|
|
|
|
|
|
|
|
// 菜单由 launchPage 加载并写入 menuStore,此处读缓存
|
|
|
|
|
|
const cachedMenu = menuStore.getMobileMenu;
|
|
|
|
|
|
if (cachedMenu?.length > 0) {
|
|
|
|
|
|
menuItems.value = flattenMenuToItems(cachedMenu);
|
|
|
|
|
|
} else if (userStore.getToken) {
|
|
|
|
|
|
// 首次注册登录场景:launchPage 可能因 token 未就绪返回空菜单,此处补拉
|
|
|
|
|
|
try {
|
|
|
|
|
|
const r = await getMobileMenuApi();
|
|
|
|
|
|
if (r?.result && Array.isArray(r.result) && r.result.length > 0) {
|
|
|
|
|
|
menuStore.setMobileMenu(r.result);
|
|
|
|
|
|
menuItems.value = flattenMenuToItems(r.result);
|
|
|
|
|
|
}
|
|
|
|
|
|
} catch (_e) {}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-07-21 20:21:57 +08:00
|
|
|
|
// 确保有学生年级信息才获取通知公告
|
|
|
|
|
|
if (curXs.value && curXs.value.njmcId) {
|
|
|
|
|
|
getArticleList();
|
|
|
|
|
|
} else {
|
|
|
|
|
|
setTimeout(() => {
|
|
|
|
|
|
if (curXs.value && curXs.value.njmcId) {
|
|
|
|
|
|
getArticleList();
|
|
|
|
|
|
}
|
|
|
|
|
|
}, 100);
|
|
|
|
|
|
}
|
2025-09-15 23:28:07 +08:00
|
|
|
|
|
2025-08-04 15:23:50 +08:00
|
|
|
|
await checkSubscribeStatus();
|
2025-06-14 22:28:04 +08:00
|
|
|
|
});
|
2025-07-21 20:21:57 +08:00
|
|
|
|
|
2025-08-04 15:23:50 +08:00
|
|
|
|
// 检查关注状态
|
|
|
|
|
|
const checkSubscribeStatus = async () => {
|
|
|
|
|
|
const userInfo = userStore.getUser;
|
2025-09-15 23:28:07 +08:00
|
|
|
|
|
2025-08-04 15:23:50 +08:00
|
|
|
|
if (userInfo && !userInfo.subscribed) {
|
|
|
|
|
|
// 延迟显示关注提醒,避免与启动页面冲突
|
|
|
|
|
|
setTimeout(() => {
|
|
|
|
|
|
showSubscribeReminder();
|
|
|
|
|
|
}, 2000);
|
|
|
|
|
|
}
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
// 显示关注提醒
|
|
|
|
|
|
const showSubscribeReminder = () => {
|
|
|
|
|
|
uni.showModal({
|
|
|
|
|
|
title: '关注服务号',
|
|
|
|
|
|
content: '请先关注我们的服务号,以便接收重要通知',
|
|
|
|
|
|
showCancel: true,
|
|
|
|
|
|
cancelText: '稍后关注',
|
|
|
|
|
|
confirmText: '立即关注',
|
|
|
|
|
|
success: (res) => {
|
|
|
|
|
|
if (res.confirm) {
|
|
|
|
|
|
// 跳转到关注页面
|
|
|
|
|
|
uni.navigateTo({
|
|
|
|
|
|
url: '/pages/system/subscribe/index'
|
|
|
|
|
|
});
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
});
|
|
|
|
|
|
};
|
|
|
|
|
|
|
2025-07-21 20:21:57 +08:00
|
|
|
|
// 监听学生信息变化,当学生信息更新时重新获取通知公告
|
|
|
|
|
|
watch(curXs, (newXs, oldXs) => {
|
|
|
|
|
|
if (newXs && newXs.njmcId) {
|
|
|
|
|
|
getArticleList();
|
|
|
|
|
|
}
|
|
|
|
|
|
}, { immediate: false });
|
2025-04-30 01:43:23 +08:00
|
|
|
|
</script>
|
|
|
|
|
|
|
|
|
|
|
|
<style lang="scss" scoped>
|
|
|
|
|
|
.home-page {
|
2025-05-30 17:22:30 +08:00
|
|
|
|
background-color: #ffffff;
|
2025-04-30 01:43:23 +08:00
|
|
|
|
min-height: 100vh;
|
|
|
|
|
|
position: relative;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.content-container {
|
|
|
|
|
|
padding: 15px;
|
2025-05-30 17:22:30 +08:00
|
|
|
|
position: relative;
|
|
|
|
|
|
z-index: 1;
|
2025-04-30 01:43:23 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* 用户信息卡片 */
|
|
|
|
|
|
.user-info-card {
|
2025-05-30 17:22:30 +08:00
|
|
|
|
position: relative;
|
|
|
|
|
|
background: linear-gradient(135deg, #ffffff 0%, #f8faff 100%);
|
|
|
|
|
|
border-radius: 20px;
|
|
|
|
|
|
padding: 20px;
|
|
|
|
|
|
margin-bottom: 20px;
|
|
|
|
|
|
margin-top: 20px;
|
|
|
|
|
|
box-shadow: 0 12px 40px rgba(74, 144, 226, 0.25);
|
|
|
|
|
|
border: 1px solid rgba(255, 255, 255, 0.8);
|
|
|
|
|
|
overflow: hidden;
|
2025-06-23 18:20:19 +08:00
|
|
|
|
|
|
|
|
|
|
.banner-content {
|
2025-09-15 23:28:07 +08:00
|
|
|
|
position: absolute;
|
|
|
|
|
|
top: 0;
|
|
|
|
|
|
bottom: 0;
|
|
|
|
|
|
left: 0;
|
|
|
|
|
|
right: 0;
|
|
|
|
|
|
|
2025-06-23 18:20:19 +08:00
|
|
|
|
.banner-img {
|
|
|
|
|
|
width: 100%;
|
|
|
|
|
|
height: 100%;
|
|
|
|
|
|
object-fit: cover;
|
|
|
|
|
|
}
|
2025-09-15 23:28:07 +08:00
|
|
|
|
|
2025-06-23 18:20:19 +08:00
|
|
|
|
.banner-overlay {
|
|
|
|
|
|
position: absolute;
|
|
|
|
|
|
top: 0;
|
|
|
|
|
|
left: 0;
|
|
|
|
|
|
width: 100%;
|
|
|
|
|
|
height: 100%;
|
|
|
|
|
|
background: linear-gradient(135deg, rgba(74, 144, 226, 0.8) 0%, rgba(53, 122, 189, 0.9) 100%);
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
align-items: center;
|
|
|
|
|
|
justify-content: center;
|
2025-09-15 23:28:07 +08:00
|
|
|
|
|
2025-06-23 18:20:19 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-06-29 19:24:41 +08:00
|
|
|
|
.glxs-btn {
|
|
|
|
|
|
position: absolute;
|
|
|
|
|
|
right: 0;
|
|
|
|
|
|
bottom: 0;
|
2025-05-30 17:22:30 +08:00
|
|
|
|
display: flex;
|
|
|
|
|
|
align-items: center;
|
2025-09-15 23:28:07 +08:00
|
|
|
|
justify-content: center;
|
2025-06-29 19:24:41 +08:00
|
|
|
|
gap: 4px;
|
|
|
|
|
|
padding: 8px 16px;
|
|
|
|
|
|
background: linear-gradient(135deg, #4A90E2 0%, #357ABD 100%);
|
|
|
|
|
|
color: #ffffff;
|
|
|
|
|
|
border-top-left-radius: 20px;
|
|
|
|
|
|
border-bottom-right-radius: 20px;
|
|
|
|
|
|
font-size: 13px;
|
|
|
|
|
|
font-weight: 500;
|
|
|
|
|
|
box-shadow: 0 4px 12px rgba(74, 144, 226, 0.3);
|
|
|
|
|
|
transition: all 0.3s ease;
|
|
|
|
|
|
flex-shrink: 0;
|
|
|
|
|
|
min-width: 60px;
|
|
|
|
|
|
height: 16px;
|
2025-09-15 23:28:07 +08:00
|
|
|
|
|
2025-06-29 19:24:41 +08:00
|
|
|
|
text {
|
|
|
|
|
|
line-height: 1;
|
2025-04-30 01:43:23 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* 功能菜单 */
|
2025-05-30 17:22:30 +08:00
|
|
|
|
.menu-section {
|
|
|
|
|
|
margin-bottom: 25px;
|
2025-09-15 23:28:07 +08:00
|
|
|
|
|
2025-05-30 17:22:30 +08:00
|
|
|
|
.section-title {
|
2025-04-30 01:43:23 +08:00
|
|
|
|
display: flex;
|
|
|
|
|
|
align-items: center;
|
2025-05-30 17:22:30 +08:00
|
|
|
|
margin-bottom: 20px;
|
|
|
|
|
|
padding: 0 5px;
|
2025-09-15 23:28:07 +08:00
|
|
|
|
|
2025-05-30 17:22:30 +08:00
|
|
|
|
.title-text {
|
|
|
|
|
|
font-size: 18px;
|
|
|
|
|
|
font-weight: 600;
|
2025-04-30 01:43:23 +08:00
|
|
|
|
color: #303133;
|
2025-05-30 17:22:30 +08:00
|
|
|
|
position: relative;
|
|
|
|
|
|
padding-left: 12px;
|
2025-09-15 23:28:07 +08:00
|
|
|
|
|
2025-05-30 17:22:30 +08:00
|
|
|
|
&::before {
|
|
|
|
|
|
content: '';
|
|
|
|
|
|
position: absolute;
|
|
|
|
|
|
left: 0;
|
|
|
|
|
|
top: 50%;
|
|
|
|
|
|
transform: translateY(-50%);
|
|
|
|
|
|
width: 4px;
|
|
|
|
|
|
height: 16px;
|
|
|
|
|
|
background: linear-gradient(135deg, #4A90E2 0%, #357ABD 100%);
|
|
|
|
|
|
border-radius: 2px;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2025-09-15 23:28:07 +08:00
|
|
|
|
|
2025-05-30 17:22:30 +08:00
|
|
|
|
.title-line {
|
|
|
|
|
|
flex: 1;
|
|
|
|
|
|
height: 1px;
|
|
|
|
|
|
background: linear-gradient(90deg, #e0e6ed 0%, transparent 100%);
|
|
|
|
|
|
margin-left: 20px;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2025-09-15 23:28:07 +08:00
|
|
|
|
|
2025-05-30 17:22:30 +08:00
|
|
|
|
.grid-menu {
|
|
|
|
|
|
display: grid;
|
|
|
|
|
|
grid-template-columns: repeat(3, 1fr);
|
|
|
|
|
|
gap: 0;
|
|
|
|
|
|
background: linear-gradient(135deg, #ffffff 0%, #f8faff 100%);
|
|
|
|
|
|
border-radius: 16px;
|
|
|
|
|
|
box-shadow: 0 8px 32px rgba(74, 144, 226, 0.2);
|
|
|
|
|
|
border: 1px solid rgba(255, 255, 255, 0.8);
|
2025-09-15 23:28:07 +08:00
|
|
|
|
|
2025-05-30 17:22:30 +08:00
|
|
|
|
.grid-item {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
flex-direction: column;
|
|
|
|
|
|
align-items: center;
|
|
|
|
|
|
justify-content: center;
|
|
|
|
|
|
padding: 20px 10px;
|
|
|
|
|
|
transition: all 0.3s ease;
|
|
|
|
|
|
border-radius: 12px;
|
|
|
|
|
|
position: relative;
|
2025-09-15 23:28:07 +08:00
|
|
|
|
|
2025-05-30 17:22:30 +08:00
|
|
|
|
&:active {
|
|
|
|
|
|
transform: scale(0.95);
|
|
|
|
|
|
background-color: rgba(74, 144, 226, 0.05);
|
|
|
|
|
|
}
|
2025-09-15 23:28:07 +08:00
|
|
|
|
|
2025-05-30 17:22:30 +08:00
|
|
|
|
.grid-icon-container {
|
|
|
|
|
|
position: relative;
|
|
|
|
|
|
width: 48px;
|
|
|
|
|
|
height: 48px;
|
|
|
|
|
|
margin-bottom: 12px;
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
align-items: center;
|
|
|
|
|
|
justify-content: center;
|
2025-09-15 23:28:07 +08:00
|
|
|
|
|
2025-05-30 17:22:30 +08:00
|
|
|
|
.icon-background {
|
|
|
|
|
|
position: absolute;
|
|
|
|
|
|
top: 0;
|
|
|
|
|
|
left: 0;
|
|
|
|
|
|
width: 100%;
|
|
|
|
|
|
height: 100%;
|
|
|
|
|
|
border-radius: 12px;
|
|
|
|
|
|
background: linear-gradient(135deg, #f0f4ff 0%, #e6f0ff 100%);
|
|
|
|
|
|
box-shadow: 0 2px 8px rgba(74, 144, 226, 0.15);
|
|
|
|
|
|
}
|
2025-09-15 23:28:07 +08:00
|
|
|
|
|
2025-05-30 17:22:30 +08:00
|
|
|
|
.grid-icon {
|
|
|
|
|
|
position: relative;
|
|
|
|
|
|
z-index: 1;
|
|
|
|
|
|
width: 28px;
|
|
|
|
|
|
height: 28px;
|
|
|
|
|
|
object-fit: contain;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2025-09-15 23:28:07 +08:00
|
|
|
|
|
2025-05-30 17:22:30 +08:00
|
|
|
|
.grid-text {
|
|
|
|
|
|
font-size: 13px;
|
|
|
|
|
|
font-weight: 500;
|
|
|
|
|
|
color: #303133;
|
|
|
|
|
|
text-align: center;
|
|
|
|
|
|
line-height: 1.2;
|
|
|
|
|
|
}
|
2025-04-30 01:43:23 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* 通知公告 */
|
|
|
|
|
|
.notice-section {
|
2025-05-30 17:22:30 +08:00
|
|
|
|
.section-title {
|
2025-04-30 01:43:23 +08:00
|
|
|
|
display: flex;
|
|
|
|
|
|
align-items: center;
|
2025-05-30 17:22:30 +08:00
|
|
|
|
margin-bottom: 20px;
|
|
|
|
|
|
padding: 0 5px;
|
2025-09-15 23:28:07 +08:00
|
|
|
|
|
2025-05-30 17:22:30 +08:00
|
|
|
|
.title-text {
|
|
|
|
|
|
font-size: 18px;
|
|
|
|
|
|
font-weight: 600;
|
2025-04-30 01:43:23 +08:00
|
|
|
|
color: #303133;
|
2025-05-30 17:22:30 +08:00
|
|
|
|
position: relative;
|
|
|
|
|
|
padding-left: 12px;
|
2025-09-15 23:28:07 +08:00
|
|
|
|
|
2025-05-30 17:22:30 +08:00
|
|
|
|
&::before {
|
|
|
|
|
|
content: '';
|
|
|
|
|
|
position: absolute;
|
|
|
|
|
|
left: 0;
|
|
|
|
|
|
top: 50%;
|
|
|
|
|
|
transform: translateY(-50%);
|
|
|
|
|
|
width: 4px;
|
|
|
|
|
|
height: 16px;
|
|
|
|
|
|
background: linear-gradient(135deg, #4A90E2 0%, #357ABD 100%);
|
|
|
|
|
|
border-radius: 2px;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2025-09-15 23:28:07 +08:00
|
|
|
|
|
2025-05-30 17:22:30 +08:00
|
|
|
|
.title-line {
|
|
|
|
|
|
flex: 1;
|
|
|
|
|
|
height: 1px;
|
|
|
|
|
|
background: linear-gradient(90deg, #e0e6ed 0%, transparent 100%);
|
|
|
|
|
|
margin-left: 20px;
|
2025-04-30 01:43:23 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
2025-09-15 23:28:07 +08:00
|
|
|
|
|
2025-04-30 01:43:23 +08:00
|
|
|
|
.notice-list {
|
|
|
|
|
|
.notice-item {
|
|
|
|
|
|
display: flex;
|
2025-05-30 17:22:30 +08:00
|
|
|
|
align-items: flex-start;
|
|
|
|
|
|
background: linear-gradient(135deg, #ffffff 0%, #f8faff 100%);
|
|
|
|
|
|
padding: 18px;
|
|
|
|
|
|
margin-bottom: 12px;
|
|
|
|
|
|
border-radius: 16px;
|
|
|
|
|
|
box-shadow: 0 8px 24px rgba(74, 144, 226, 0.18);
|
|
|
|
|
|
border: 1px solid rgba(255, 255, 255, 0.8);
|
|
|
|
|
|
transition: all 0.3s ease;
|
2025-09-15 23:28:07 +08:00
|
|
|
|
|
2025-05-30 17:22:30 +08:00
|
|
|
|
&:active {
|
|
|
|
|
|
transform: translateY(-2px);
|
|
|
|
|
|
box-shadow: 0 12px 36px rgba(74, 144, 226, 0.25);
|
2025-04-30 01:43:23 +08:00
|
|
|
|
}
|
2025-09-15 23:28:07 +08:00
|
|
|
|
|
2025-05-30 17:22:30 +08:00
|
|
|
|
.notice-icon {
|
|
|
|
|
|
width: 50px;
|
|
|
|
|
|
height: 50px;
|
|
|
|
|
|
background: linear-gradient(135deg, #f0f4ff 0%, #e6f0ff 100%);
|
|
|
|
|
|
border-radius: 12px;
|
|
|
|
|
|
margin-right: 15px;
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
align-items: center;
|
|
|
|
|
|
justify-content: center;
|
|
|
|
|
|
box-shadow: 0 2px 8px rgba(74, 144, 226, 0.15);
|
|
|
|
|
|
flex-shrink: 0;
|
|
|
|
|
|
}
|
2025-09-15 23:28:07 +08:00
|
|
|
|
|
2025-04-30 01:43:23 +08:00
|
|
|
|
.notice-content {
|
|
|
|
|
|
flex: 1;
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
flex-direction: column;
|
2025-05-30 17:22:30 +08:00
|
|
|
|
min-width: 0;
|
2025-09-15 23:28:07 +08:00
|
|
|
|
|
2025-04-30 01:43:23 +08:00
|
|
|
|
.notice-title {
|
|
|
|
|
|
font-size: 15px;
|
2025-05-30 17:22:30 +08:00
|
|
|
|
font-weight: 600;
|
2025-04-30 01:43:23 +08:00
|
|
|
|
color: #303133;
|
2025-05-30 17:22:30 +08:00
|
|
|
|
margin-bottom: 8px;
|
2025-04-30 01:43:23 +08:00
|
|
|
|
overflow: hidden;
|
|
|
|
|
|
text-overflow: ellipsis;
|
|
|
|
|
|
display: -webkit-box;
|
|
|
|
|
|
-webkit-line-clamp: 1;
|
|
|
|
|
|
-webkit-box-orient: vertical;
|
2025-05-30 17:22:30 +08:00
|
|
|
|
line-height: 1.3;
|
2025-04-30 01:43:23 +08:00
|
|
|
|
}
|
2025-09-15 23:28:07 +08:00
|
|
|
|
|
2025-04-30 01:43:23 +08:00
|
|
|
|
.notice-desc {
|
|
|
|
|
|
font-size: 13px;
|
2025-05-30 17:22:30 +08:00
|
|
|
|
color: #606266;
|
|
|
|
|
|
margin-bottom: 12px;
|
2025-04-30 01:43:23 +08:00
|
|
|
|
overflow: hidden;
|
|
|
|
|
|
text-overflow: ellipsis;
|
|
|
|
|
|
display: -webkit-box;
|
|
|
|
|
|
-webkit-line-clamp: 2;
|
|
|
|
|
|
-webkit-box-orient: vertical;
|
|
|
|
|
|
flex: 1;
|
2025-05-30 17:22:30 +08:00
|
|
|
|
line-height: 1.4;
|
2025-04-30 01:43:23 +08:00
|
|
|
|
}
|
2025-09-15 23:28:07 +08:00
|
|
|
|
|
2025-05-30 17:22:30 +08:00
|
|
|
|
.notice-footer {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
align-items: center;
|
|
|
|
|
|
justify-content: space-between;
|
2025-09-15 23:28:07 +08:00
|
|
|
|
|
2025-05-30 17:22:30 +08:00
|
|
|
|
.notice-date {
|
|
|
|
|
|
font-size: 12px;
|
|
|
|
|
|
color: #909399;
|
|
|
|
|
|
font-weight: 400;
|
|
|
|
|
|
line-height: 1;
|
|
|
|
|
|
}
|
2025-09-15 23:28:07 +08:00
|
|
|
|
|
2025-05-30 17:22:30 +08:00
|
|
|
|
.notice-arrow {
|
|
|
|
|
|
width: 20px;
|
|
|
|
|
|
height: 20px;
|
|
|
|
|
|
background-color: #f5f7fa;
|
|
|
|
|
|
border-radius: 50%;
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
align-items: center;
|
|
|
|
|
|
justify-content: center;
|
|
|
|
|
|
flex-shrink: 0;
|
|
|
|
|
|
}
|
2025-04-30 01:43:23 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-05-30 17:22:30 +08:00
|
|
|
|
/* 响应式优化 */
|
|
|
|
|
|
@media (max-width: 375px) {
|
|
|
|
|
|
.content-container {
|
|
|
|
|
|
padding: 12px;
|
|
|
|
|
|
}
|
2025-06-29 19:24:41 +08:00
|
|
|
|
|
2025-05-30 17:22:30 +08:00
|
|
|
|
.grid-menu .grid-item {
|
|
|
|
|
|
padding: 15px 8px;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2025-04-30 01:43:23 +08:00
|
|
|
|
</style>
|