mirror of
https://gitee.com/myxzgzs/boyuehasfj-vue3-html.git
synced 2025-08-07 22:52:42 +08:00
460 lines
12 KiB
Vue
460 lines
12 KiB
Vue
<script setup lang="ts">
|
|
import { ref, onMounted, watch, onActivated } from 'vue'
|
|
import { getPageList } from '../services/api'
|
|
import TheNavbar from '../components/TheNavbar.vue'
|
|
import Pagination from '../components/Pagination.vue'
|
|
|
|
interface PageItem {
|
|
id: number;
|
|
formatId: string;
|
|
title: string;
|
|
content: string;
|
|
pageType: string;
|
|
pageUrl?: string;
|
|
createTime: string;
|
|
updateTime: string;
|
|
status: number;
|
|
sortOrder: number;
|
|
viewCount: number;
|
|
author?: string;
|
|
keywords?: string;
|
|
description?: string;
|
|
attachmentUrl?: string;
|
|
}
|
|
|
|
interface PageData {
|
|
code: number;
|
|
rows: PageItem[];
|
|
total: number;
|
|
msg: string;
|
|
}
|
|
|
|
const laws = ref<PageItem[]>([]);
|
|
const loading = ref(true);
|
|
const error = ref('');
|
|
const total = ref(0);
|
|
const currentPage = ref(1);
|
|
const pageSize = ref(15);
|
|
|
|
const fetchLaws = async () => {
|
|
try {
|
|
loading.value = true;
|
|
error.value = '';
|
|
|
|
const response = await getPageList('law', currentPage.value, pageSize.value);
|
|
|
|
if (response.code === 200) {
|
|
laws.value = response.rows || [];
|
|
total.value = response.total || laws.value.length;
|
|
|
|
// 如果当前页没有数据且不是第一页,尝试请求前一页
|
|
if (laws.value.length === 0 && currentPage.value > 1) {
|
|
currentPage.value = currentPage.value - 1;
|
|
await fetchLaws();
|
|
}
|
|
} else {
|
|
error.value = response.msg || '获取数据失败';
|
|
laws.value = [];
|
|
}
|
|
} catch (err) {
|
|
console.error('获取法律规定列表失败:', err);
|
|
error.value = '获取数据失败,请稍后重试';
|
|
laws.value = [];
|
|
} finally {
|
|
loading.value = false;
|
|
}
|
|
};
|
|
|
|
// 页码或每页数量变化时重新获取数据
|
|
watch([currentPage, pageSize], () => {
|
|
fetchLaws();
|
|
});
|
|
|
|
// 分页变化处理
|
|
const handlePageChange = (page: number, size: number) => {
|
|
currentPage.value = page;
|
|
pageSize.value = size;
|
|
};
|
|
|
|
// 格式化日期
|
|
const formatDate = (dateStr: string | null | undefined): string => {
|
|
if (!dateStr) return '未知日期';
|
|
try {
|
|
const date = new Date(dateStr);
|
|
if (isNaN(date.getTime())) {
|
|
return '未知日期';
|
|
}
|
|
return date.toLocaleDateString('zh-CN', {
|
|
year: 'numeric',
|
|
month: '2-digit',
|
|
day: '2-digit'
|
|
});
|
|
} catch (e) {
|
|
console.error('日期格式化错误:', e);
|
|
return '未知日期';
|
|
}
|
|
}
|
|
|
|
onMounted(() => {
|
|
fetchLaws();
|
|
});
|
|
|
|
// 添加页面激活时重新获取数据的处理函数
|
|
onActivated(() => {
|
|
console.log('法律规定列表页面被激活,重新获取数据');
|
|
fetchLaws();
|
|
});
|
|
</script>
|
|
|
|
<template>
|
|
<div class="law-list-container">
|
|
<TheNavbar />
|
|
|
|
<div class="page-header">
|
|
<div class="container">
|
|
<h1>法律规定</h1>
|
|
<p class="subtitle">查看并了解最新的法律法规内容</p>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="container">
|
|
<div class="content-wrapper">
|
|
<div class="law-list">
|
|
<div v-if="loading" class="loading">
|
|
<div class="spinner"></div>
|
|
<p>加载中...</p>
|
|
</div>
|
|
|
|
<div v-else-if="error" class="error">
|
|
<h3>获取数据失败</h3>
|
|
<p>{{ error }}</p>
|
|
<button @click="fetchLaws" class="btn-retry">重试</button>
|
|
</div>
|
|
|
|
<div v-else-if="laws.length === 0" class="empty">
|
|
<h3>暂无内容</h3>
|
|
<p>当前没有法律规定相关内容</p>
|
|
<router-link to="/" class="btn-home">返回首页</router-link>
|
|
</div>
|
|
|
|
<div v-else class="list-content">
|
|
<div class="law-grid">
|
|
<div v-for="law in laws" :key="law.id" class="law-card">
|
|
<router-link :to="`/show.html?Id=${law.formatId}`" class="law-link">
|
|
<div class="law-icon"></div>
|
|
<div class="law-content">
|
|
<h3 class="law-title">{{ law.title }}</h3>
|
|
<div class="law-meta">
|
|
<span class="law-date">{{ formatDate(law.createTime) }}</span>
|
|
<!-- <span class="law-views">浏览: {{ law.viewCount }}</span> -->
|
|
</div>
|
|
<div class="law-footer">
|
|
<span class="view-more">查看详情</span>
|
|
</div>
|
|
</div>
|
|
</router-link>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- 分页组件 -->
|
|
<Pagination
|
|
:total="total"
|
|
v-model:current-page="currentPage"
|
|
v-model:page-size="pageSize"
|
|
@change="handlePageChange"
|
|
:page-sizes="[8, 16, 24, 32]"
|
|
/>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</template>
|
|
|
|
<style scoped>
|
|
.law-list-container {
|
|
min-height: 100vh;
|
|
background-color: var(--color-background);
|
|
display: flex;
|
|
flex-direction: column;
|
|
width: 100%;
|
|
}
|
|
|
|
.page-header {
|
|
background-color: var(--color-primary);
|
|
color: white;
|
|
padding: 3rem 0;
|
|
margin-bottom: 0;
|
|
position: relative;
|
|
overflow: hidden;
|
|
background-image: linear-gradient(135deg, #0072ff 0%, #00c6ff 100%);
|
|
width: 100%;
|
|
}
|
|
|
|
.page-header::before {
|
|
content: '';
|
|
position: absolute;
|
|
top: 0;
|
|
left: 0;
|
|
right: 0;
|
|
bottom: 0;
|
|
background: url('');
|
|
opacity: 0.4;
|
|
z-index: 1;
|
|
}
|
|
|
|
.page-header .container {
|
|
position: relative;
|
|
z-index: 2;
|
|
max-width: 1200px;
|
|
margin: 0 auto;
|
|
padding: 0 2rem;
|
|
}
|
|
|
|
.page-header h1 {
|
|
font-size: 2.5rem;
|
|
margin-bottom: 1rem;
|
|
font-weight: 700;
|
|
letter-spacing: 2px;
|
|
text-shadow: 0 2px 4px rgba(0, 0, 0, 0.1);
|
|
}
|
|
|
|
.subtitle {
|
|
font-size: 1.2rem;
|
|
opacity: 0.9;
|
|
max-width: 700px;
|
|
line-height: 1.4;
|
|
text-shadow: 0 1px 2px rgba(0, 0, 0, 0.1);
|
|
}
|
|
|
|
.container {
|
|
max-width: 1200px;
|
|
margin: 0 auto;
|
|
padding: 0 1rem;
|
|
width: 100%;
|
|
}
|
|
|
|
.content-wrapper {
|
|
padding: 2rem 0;
|
|
width: 100%;
|
|
}
|
|
|
|
.law-list {
|
|
background-color: white;
|
|
border-radius: 12px;
|
|
padding: 2rem;
|
|
box-shadow: 0 4px 12px rgba(0, 0, 0, 0.05);
|
|
width: 100%;
|
|
}
|
|
|
|
.loading, .error, .empty {
|
|
display: flex;
|
|
flex-direction: column;
|
|
align-items: center;
|
|
justify-content: center;
|
|
padding: 3rem;
|
|
text-align: center;
|
|
}
|
|
|
|
.spinner {
|
|
width: 40px;
|
|
height: 40px;
|
|
border: 3px solid rgba(0, 0, 0, 0.1);
|
|
border-radius: 50%;
|
|
border-top-color: var(--color-primary);
|
|
animation: spin 1s ease-in-out infinite;
|
|
margin-bottom: 1rem;
|
|
}
|
|
|
|
@keyframes spin {
|
|
to { transform: rotate(360deg); }
|
|
}
|
|
|
|
.btn-retry, .btn-home {
|
|
background-color: var(--color-primary);
|
|
color: white;
|
|
border: none;
|
|
padding: 0.7rem 1.8rem;
|
|
border-radius: 4px;
|
|
cursor: pointer;
|
|
font-size: 0.95rem;
|
|
transition: all 0.3s;
|
|
margin-top: 1rem;
|
|
box-shadow: 0 2px 8px rgba(0, 123, 255, 0.25);
|
|
text-decoration: none;
|
|
display: inline-block;
|
|
}
|
|
|
|
.btn-retry:hover, .btn-home:hover {
|
|
background-color: var(--color-primary-dark);
|
|
transform: translateY(-2px);
|
|
box-shadow: 0 4px 12px rgba(0, 123, 255, 0.3);
|
|
}
|
|
|
|
.law-grid {
|
|
display: grid;
|
|
grid-template-columns: repeat(auto-fill, minmax(250px, 1fr));
|
|
gap: 1rem;
|
|
margin-bottom: 2rem;
|
|
}
|
|
|
|
.law-card {
|
|
background-color: white;
|
|
border-radius: 8px;
|
|
box-shadow: 0 2px 8px rgba(0, 0, 0, 0.08);
|
|
transition: all 0.3s ease;
|
|
overflow: hidden;
|
|
border: 1px solid #f0f0f0;
|
|
height: 100%;
|
|
}
|
|
|
|
.law-card:hover {
|
|
transform: translateY(-5px);
|
|
box-shadow: 0 8px 15px rgba(0, 0, 0, 0.1);
|
|
}
|
|
|
|
.law-link {
|
|
display: flex;
|
|
flex-direction: column;
|
|
height: 100%;
|
|
text-decoration: none;
|
|
color: var(--color-text);
|
|
padding: 1.2rem;
|
|
}
|
|
|
|
.law-icon {
|
|
width: 48px;
|
|
height: 48px;
|
|
background-color: rgba(0, 123, 255, 0.1);
|
|
border-radius: 50%;
|
|
margin-bottom: 1.2rem;
|
|
background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' fill='%230072ff' viewBox='0 0 16 16'%3E%3Cpath d='M8 15A7 7 0 1 1 8 1a7 7 0 0 1 0 14zm0 1A8 8 0 1 0 8 0a8 8 0 0 0 0 16z'/%3E%3Cpath d='M5.255 5.786a.237.237 0 0 0 .241.247h.825c.138 0 .248-.113.266-.25.09-.656.54-1.134 1.342-1.134.686 0 1.314.343 1.314 1.168 0 .635-.374.927-.965 1.371-.673.489-1.206 1.06-1.168 1.987l.003.217a.25.25 0 0 0 .25.246h.811a.25.25 0 0 0 .25-.25v-.105c0-.718.273-.927 1.01-1.486.609-.463 1.244-.977 1.244-2.056 0-1.511-1.276-2.241-2.673-2.241-1.267 0-2.655.59-2.75 2.286zm1.557 5.763c0 .533.425.927 1.01.927.609 0 1.028-.394 1.028-.927 0-.552-.42-.94-1.029-.94-.584 0-1.009.388-1.009.94z'/%3E%3C/svg%3E");
|
|
background-size: 24px;
|
|
background-position: center;
|
|
background-repeat: no-repeat;
|
|
transition: all 0.3s;
|
|
}
|
|
|
|
.law-card:hover .law-icon {
|
|
background-color: var(--color-primary);
|
|
background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' fill='white' viewBox='0 0 16 16'%3E%3Cpath d='M8 15A7 7 0 1 1 8 1a7 7 0 0 1 0 14zm0 1A8 8 0 1 0 8 0a8 8 0 0 0 0 16z'/%3E%3Cpath d='M5.255 5.786a.237.237 0 0 0 .241.247h.825c.138 0 .248-.113.266-.25.09-.656.54-1.134 1.342-1.134.686 0 1.314.343 1.314 1.168 0 .635-.374.927-.965 1.371-.673.489-1.206 1.06-1.168 1.987l.003.217a.25.25 0 0 0 .25.246h.811a.25.25 0 0 0 .25-.25v-.105c0-.718.273-.927 1.01-1.486.609-.463 1.244-.977 1.244-2.056 0-1.511-1.276-2.241-2.673-2.241-1.267 0-2.655.59-2.75 2.286zm1.557 5.763c0 .533.425.927 1.01.927.609 0 1.028-.394 1.028-.927 0-.552-.42-.94-1.029-.94-.584 0-1.009.388-1.009.94z'/%3E%3C/svg%3E");
|
|
}
|
|
|
|
.law-content {
|
|
flex: 1;
|
|
display: flex;
|
|
flex-direction: column;
|
|
}
|
|
|
|
.law-title {
|
|
font-size: 1.1rem;
|
|
font-weight: 600;
|
|
margin-bottom: 1rem;
|
|
line-height: 1.4;
|
|
color: var(--color-text);
|
|
}
|
|
|
|
.law-meta {
|
|
margin-bottom: 1.5rem;
|
|
color: var(--color-text-light);
|
|
font-size: 0.85rem;
|
|
display: flex;
|
|
flex-direction: column;
|
|
gap: 0.5rem;
|
|
}
|
|
|
|
.law-date, .law-views {
|
|
display: flex;
|
|
align-items: center;
|
|
gap: 0.5rem;
|
|
}
|
|
|
|
.law-date::before, .law-views::before {
|
|
content: '';
|
|
display: inline-block;
|
|
width: 16px;
|
|
height: 16px;
|
|
background-size: contain;
|
|
background-repeat: no-repeat;
|
|
background-position: center;
|
|
opacity: 0.7;
|
|
}
|
|
|
|
.law-date::before {
|
|
background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='16' height='16' fill='currentColor' viewBox='0 0 16 16'%3E%3Cpath d='M4.684 11.523v-2.3h2.261v-.61H4.684V6.801h2.464v-.61H4v5.332h.684zm3.296 0h.676V8.98c0-.554.227-1.007.953-1.007.125 0 .258.004.329.015v-.613a1.806 1.806 0 0 0-.254-.02c-.582 0-.891.32-1.012.567h-.02v-.504H7.98v4.105zm2.805-5.093c0 .238.192.425.43.425a.428.428 0 1 0 0-.855.426.426 0 0 0-.43.43zm.094 5.093h.672V7.418h-.672v4.105z'/%3E%3Cpath d='M3.5 0a.5.5 0 0 1 .5.5V1h8V.5a.5.5 0 0 1 1 0V1h1a2 2 0 0 1 2 2v11a2 2 0 0 1-2 2H2a2 2 0 0 1-2-2V3a2 2 0 0 1 2-2h1V.5a.5.5 0 0 1 .5-.5zM1 4v10a1 1 0 0 0 1 1h12a1 1 0 0 0 1-1V4H1z'/%3E%3C/svg%3E");
|
|
}
|
|
|
|
.law-views::before {
|
|
background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='16' height='16' fill='currentColor' viewBox='0 0 16 16'%3E%3Cpath d='M10.5 8a2.5 2.5 0 1 1-5 0 2.5 2.5 0 0 1 5 0z'/%3E%3Cpath d='M0 8s3-5.5 8-5.5S16 8 16 8s-3 5.5-8 5.5S0 8 0 8zm8 3.5a3.5 3.5 0 1 0 0-7 3.5 3.5 0 0 0 0 7z'/%3E%3C/svg%3E");
|
|
}
|
|
|
|
.law-footer {
|
|
margin-top: auto;
|
|
padding-top: 1rem;
|
|
border-top: 1px solid rgba(0, 0, 0, 0.05);
|
|
}
|
|
|
|
.view-more {
|
|
display: inline-flex;
|
|
align-items: center;
|
|
color: var(--color-primary);
|
|
font-size: 0.9rem;
|
|
transition: all 0.3s;
|
|
}
|
|
|
|
.view-more::after {
|
|
content: '→';
|
|
margin-left: 0.3rem;
|
|
transition: transform 0.3s;
|
|
}
|
|
|
|
.law-card:hover .view-more::after {
|
|
transform: translateX(3px);
|
|
}
|
|
|
|
@media (max-width: 768px) {
|
|
.page-header {
|
|
padding: 2.5rem 0;
|
|
}
|
|
|
|
.page-header h1 {
|
|
font-size: 2.2rem;
|
|
}
|
|
|
|
.law-grid {
|
|
grid-template-columns: repeat(auto-fill, minmax(220px, 1fr));
|
|
gap: 1rem;
|
|
}
|
|
|
|
.law-list {
|
|
padding: 1.5rem;
|
|
}
|
|
}
|
|
|
|
@media (max-width: 576px) {
|
|
.page-header {
|
|
padding: 2rem 0;
|
|
}
|
|
|
|
.page-header h1 {
|
|
font-size: 1.8rem;
|
|
}
|
|
|
|
.subtitle {
|
|
font-size: 1rem;
|
|
}
|
|
|
|
.law-grid {
|
|
grid-template-columns: 1fr;
|
|
}
|
|
|
|
.law-list {
|
|
padding: 1rem;
|
|
border-radius: 8px;
|
|
}
|
|
|
|
.law-link {
|
|
padding: 1.2rem;
|
|
}
|
|
}
|
|
</style> |