feat: 界面优化1

dev_na
puz 2026-07-02 17:29:39 +08:00
parent f5a6a22eb1
commit cc43fe2e01
22 changed files with 605 additions and 314 deletions

View File

@ -177,6 +177,13 @@
line-height: 22px;
}
.data-list-panel__table-area .ant-table-thead > tr > th:last-child,
.data-list-panel__table-area .ant-table-tbody > tr > td:last-child,
.data-list-panel__table-area .ant-table-thead > tr > th.ant-table-cell-fix-right,
.data-list-panel__table-area .ant-table-tbody > tr > td.ant-table-cell-fix-right {
padding-right: 24px;
}
.data-list-panel__table-area .ant-table-content,
.data-list-panel__table-area .ant-table-body {
overflow-x: auto !important;

View File

@ -70,6 +70,24 @@
border-bottom: 1px solid var(--app-border-color, #f0f0f0);
}
.list-table-container .ant-table-thead > tr > th:last-child,
.list-table-container .ant-table-tbody > tr > td:last-child,
.list-table-container .ant-table-thead > tr > th.ant-table-cell-fix-right,
.list-table-container .ant-table-tbody > tr > td.ant-table-cell-fix-right {
padding-right: 24px;
}
.list-table-container .ant-table-thead > tr > th:last-child,
.list-table-container .ant-table-tbody > tr > td:last-child {
text-align: right;
}
.list-table-container .ant-table-tbody > tr > td:last-child .ant-space,
.list-table-container .ant-table-tbody > tr > td.ant-table-cell-fix-right .ant-space {
justify-content: flex-end;
width: 100%;
}
.list-table-container .ant-table-tbody > tr:not(.row-selected):not(.ant-table-row-selected):hover > td {
background: var(--app-surface-color, #fff) !important;
}

View File

@ -0,0 +1,38 @@
.summary-stat-cards {
flex-shrink: 0;
min-width: 0;
}
.summary-stat-cards__card {
height: 100%;
min-height: 112px;
border: 1px solid #e6e6e6;
border-radius: 12px;
background: #fff;
box-shadow: none;
}
.summary-stat-cards__card .ant-card-body {
height: 100%;
padding: 26px 24px;
display: flex;
align-items: center;
}
.summary-stat-cards__label {
color: var(--app-text-secondary, #9095a1);
font-size: 13px;
}
.summary-stat-cards__card .ant-statistic-content {
display: flex;
align-items: center;
line-height: 1;
}
.summary-stat-cards__icon {
margin-right: 8px;
display: inline-flex;
align-items: center;
line-height: 1;
}

View File

@ -0,0 +1,41 @@
import type { CSSProperties, ReactNode } from "react";
import { Card, Col, Row, Statistic } from "antd";
import "./SummaryStatCards.css";
export interface SummaryStatCardItem {
key: string;
label: ReactNode;
value: string | number;
icon: ReactNode;
color: string;
}
interface SummaryStatCardsProps {
items: SummaryStatCardItem[];
ariaLabel?: string;
}
export default function SummaryStatCards({ items, ariaLabel }: SummaryStatCardsProps) {
return (
<div className="summary-stat-cards" aria-label={ariaLabel}>
<Row gutter={[16, 16]}>
{items.map((item) => (
<Col xs={24} sm={12} xl={6} key={item.key}>
<Card className="summary-stat-cards__card" variant="borderless">
<Statistic
title={<span className="summary-stat-cards__label">{item.label}</span>}
value={item.value}
valueStyle={{ color: item.color, fontWeight: 700 } as CSSProperties}
prefix={
<span className="summary-stat-cards__icon" style={{ color: item.color }}>
{item.icon}
</span>
}
/>
</Card>
</Col>
))}
</Row>
</div>
);
}

View File

@ -680,8 +680,18 @@ export default function Users() {
</Form.Item>
) : null}
<Row gutter={16}>
<Col span={12}><Form.Item label={t("common.status")} name="status" initialValue={1}><Select
options={statusDict.map((item) => ({label: item.itemLabel, value: Number(item.itemValue)}))}/></Form.Item></Col>
<Col span={12}>
<Form.Item
label={t("common.status")}
name="status"
initialValue={1}
valuePropName="checked"
getValueProps={(value) => ({ checked: value !== 0 })}
getValueFromEvent={(checked: boolean) => (checked ? 1 : 0)}
>
<Switch checkedChildren={t("usersExt.enabled")} unCheckedChildren={t("usersExt.disabled")} />
</Form.Item>
</Col>
{isPlatformMode &&
<Col span={12}><Form.Item label={t("users.platformAdmin")} name="isPlatformAdmin" valuePropName="checked"><Switch/></Form.Item></Col>}
</Row>

View File

@ -12,42 +12,7 @@
background: transparent;
}
.client-summary-chips {
display: flex;
align-items: center;
gap: 8px;
flex-wrap: wrap;
min-width: 0;
}
.client-summary-chip {
height: 32px;
display: inline-flex;
align-items: center;
gap: 6px;
padding: 0 10px;
border: 1px solid #e6e6e6;
border-radius: 4px;
background: #f9fafe;
color: #333;
font-size: 13px;
line-height: 20px;
white-space: nowrap;
}
.client-summary-chip .anticon {
color: #3c70f5;
}
.client-summary-chip strong {
color: #1677ff;
font-size: 16px;
font-weight: 600;
}
@media (max-width: 768px) {
.client-summary-chips,
.client-summary-chip {
width: 100%;
}
.client-management-page__content {
gap: 12px;
padding: 8px;
}

View File

@ -23,6 +23,7 @@ import DataListPanel from "@/components/shared/DataListPanel";
import ListTable from "@/components/shared/ListTable/ListTable";
import AppPagination from "@/components/shared/AppPagination";
import SectionCard from "@/components/shared/SectionCard";
import SummaryStatCards from "@/components/shared/SummaryStatCards";
import { createClientDownload, deleteClientDownload, listClientDownloads, type ClientDownloadDTO, type ClientDownloadVO, updateClientDownload, uploadClientPackage } from "@/api/business/client";
import { fetchDictItemsByTypeCode } from "@/api/dict";
import { useDict } from "@/hooks/useDict";
@ -236,6 +237,13 @@ export default function ClientManagement() {
groups: platformGroups.length,
}), [platformGroups.length, records]);
const statCards = useMemo(() => [
{ key: "total", label: "发布总数", value: stats.total, icon: <CloudUploadOutlined />, color: "#1890ff" },
{ key: "enabled", label: "已启用", value: stats.enabled, icon: <LaptopOutlined />, color: "#faad14" },
{ key: "latest", label: "最新版本", value: stats.latest, icon: <MobileOutlined />, color: "#52c41a" },
{ key: "groups", label: "平台分组", value: stats.groups, icon: <RocketOutlined />, color: "#13c2c2" },
], [stats]);
const openCreate = () => {
const firstOption = platformGroups[0]?.options[0];
if (!firstOption) {
@ -410,20 +418,14 @@ export default function ClientManagement() {
<SectionCard
title="客户端管理"
description="发布平台由数据字典 client_platform 驱动,并按父子分组展示。"
contentClassName="client-management-page__content"
>
<SummaryStatCards items={statCards} ariaLabel="客户端发布统计" />
<DataListPanel
leftActions={
<Space wrap>
<Button type="primary" icon={<PlusOutlined />} onClick={openCreate}>
</Button>
<div className="client-summary-chips" aria-label="客户端发布统计">
<span className="client-summary-chip"><CloudUploadOutlined aria-hidden="true" /><span></span><strong>{stats.total}</strong></span>
<span className="client-summary-chip"><LaptopOutlined aria-hidden="true" /><span></span><strong>{stats.enabled}</strong></span>
<span className="client-summary-chip"><MobileOutlined aria-hidden="true" /><span></span><strong>{stats.latest}</strong></span>
<span className="client-summary-chip"><RocketOutlined aria-hidden="true" /><span></span><strong>{stats.groups}</strong></span>
</div>
</Space>
<Button type="primary" icon={<PlusOutlined />} onClick={openCreate}>
</Button>
}
rightActions={
<Space wrap>

View File

@ -12,37 +12,9 @@
background: transparent;
}
.external-app-summary-chips {
display: flex;
align-items: center;
gap: 8px;
flex-wrap: wrap;
min-width: 0;
}
.external-app-summary-chip {
height: 32px;
display: inline-flex;
align-items: center;
gap: 6px;
padding: 0 10px;
border: 1px solid #e6e6e6;
border-radius: 4px;
background: #f9fafe;
color: #333;
font-size: 13px;
line-height: 20px;
white-space: nowrap;
}
.external-app-summary-chip .anticon {
color: #3c70f5;
}
.external-app-summary-chip strong {
color: #1677ff;
font-size: 16px;
font-weight: 600;
.external-app-page__content {
gap: 12px;
padding: 8px;
}
.external-app-icon-form-item {
@ -133,11 +105,6 @@
}
@media (max-width: 768px) {
.external-app-summary-chips,
.external-app-summary-chip {
width: 100%;
}
.external-app-icon-picker {
width: 100%;
min-height: 116px;

View File

@ -7,11 +7,13 @@ import DataListPanel from "@/components/shared/DataListPanel";
import ListTable from "@/components/shared/ListTable/ListTable";
import AppPagination from "@/components/shared/AppPagination";
import SectionCard from "@/components/shared/SectionCard";
import SummaryStatCards from "@/components/shared/SummaryStatCards";
import { createExternalApp, deleteExternalApp, listExternalApps, type ExternalAppDTO, type ExternalAppVO, updateExternalApp, uploadExternalAppApk, uploadExternalAppIcon } from "@/api/business/externalApp";
import "./ExternalAppManagement.css";
const { Text } = Typography;
const { TextArea } = Input;
const ICON_UPLOAD_ACCEPT = ".png,.jpg,.jpeg,image/png,image/jpeg";
type ExternalAppFormValues = {
appName: string;
@ -121,6 +123,13 @@ export default function ExternalAppManagement() {
enabled: records.filter((item) => item.status === 1).length,
}), [records]);
const statCards = useMemo(() => [
{ key: "total", label: "应用总数", value: stats.total, icon: <AppstoreOutlined />, color: "#1890ff" },
{ key: "native", label: "原生应用", value: stats.native, icon: <RobotOutlined />, color: "#faad14" },
{ key: "web", label: "Web 应用", value: stats.web, icon: <GlobalOutlined />, color: "#52c41a" },
{ key: "enabled", label: "已启用", value: stats.enabled, icon: <PictureOutlined />, color: "#13c2c2" },
], [stats]);
const openCreate = () => {
setEditing(null);
form.resetFields();
@ -218,6 +227,20 @@ export default function ExternalAppManagement() {
}
};
const beforeUploadIcon = (file: File) => {
const isAllowedIcon = file.type
? file.type === "image/png" || file.type === "image/jpeg"
: /\.(png|jpe?g)$/i.test(file.name);
if (!isAllowedIcon) {
message.warning("应用图标仅支持上传 PNG / JPG 格式");
return Upload.LIST_IGNORE;
}
void handleUploadIcon(file);
return Upload.LIST_IGNORE;
};
const handleToggleStatus = async (record: ExternalAppVO, checked: boolean) => {
await updateExternalApp(record.id, { status: checked ? 1 : 0 });
message.success(checked ? "已启用应用" : "已停用应用");
@ -299,20 +322,14 @@ export default function ExternalAppManagement() {
<SectionCard
title="外部应用管理"
description="统一维护首页九宫格与抽屉入口中的原生应用、Web 服务和应用图标资源。"
contentClassName="external-app-page__content"
>
<SummaryStatCards items={statCards} ariaLabel="外部应用统计" />
<DataListPanel
leftActions={
<Space wrap>
<Button type="primary" icon={<PlusOutlined />} onClick={openCreate}>
</Button>
<div className="external-app-summary-chips" aria-label="外部应用统计">
<span className="external-app-summary-chip"><AppstoreOutlined aria-hidden="true" /><span></span><strong>{stats.total}</strong></span>
<span className="external-app-summary-chip"><RobotOutlined aria-hidden="true" /><span></span><strong>{stats.native}</strong></span>
<span className="external-app-summary-chip"><GlobalOutlined aria-hidden="true" /><span>Web </span><strong>{stats.web}</strong></span>
<span className="external-app-summary-chip"><PictureOutlined aria-hidden="true" /><span></span><strong>{stats.enabled}</strong></span>
</div>
</Space>
<Button type="primary" icon={<PlusOutlined />} onClick={openCreate}>
</Button>
}
rightActions={
<Space wrap>
@ -360,9 +377,9 @@ export default function ExternalAppManagement() {
<Form.Item label="应用图标" className="external-app-icon-form-item">
<Upload
accept="image/png,image/jpeg,image/webp,image/svg+xml"
accept={ICON_UPLOAD_ACCEPT}
showUploadList={false}
beforeUpload={(file) => { void handleUploadIcon(file as File); return Upload.LIST_IGNORE; }}
beforeUpload={(file) => beforeUploadIcon(file as File)}
>
<button
type="button"
@ -380,7 +397,7 @@ export default function ExternalAppManagement() {
<span className="external-app-icon-picker__text">
{uploadingIcon ? "上传中..." : watchedIconUrl ? "更换图标" : "点击上传图标"}
</span>
<span className="external-app-icon-picker__hint">PNG / JPG / WebP / SVG</span>
<span className="external-app-icon-picker__hint">PNG / JPG</span>
</button>
</Upload>
</Form.Item>

View File

@ -12,42 +12,7 @@
background: transparent;
}
.license-summary-chips {
display: flex;
align-items: center;
gap: 8px;
flex-wrap: wrap;
min-width: 0;
}
.license-summary-chip {
height: 32px;
display: inline-flex;
align-items: center;
gap: 6px;
padding: 0 10px;
border: 1px solid #e6e6e6;
border-radius: 4px;
background: #f9fafe;
color: #333;
font-size: 13px;
line-height: 20px;
white-space: nowrap;
}
.license-summary-chip .anticon {
color: #3c70f5;
}
.license-summary-chip strong {
color: #1677ff;
font-size: 16px;
font-weight: 600;
}
@media (max-width: 768px) {
.license-summary-chips,
.license-summary-chip {
width: 100%;
}
.license-management-page__content {
gap: 12px;
padding: 8px;
}

View File

@ -7,6 +7,7 @@ import AppPagination from "@/components/shared/AppPagination";
import DataListPanel from "@/components/shared/DataListPanel";
import ListTable from "@/components/shared/ListTable/ListTable";
import SectionCard from "@/components/shared/SectionCard";
import SummaryStatCards from "@/components/shared/SummaryStatCards";
import { listLicenses, type LicenseVO } from "@/api/business/license";
import "./LicenseManagement.css";
@ -92,6 +93,13 @@ export default function LicenseManagement() {
available: records.filter((item) => item.licenseStatus === 1).length,
}), [records]);
const statCards = useMemo(() => [
{ key: "total", label: "授权总数", value: stats.total, icon: <KeyOutlined />, color: "#1890ff" },
{ key: "using", label: "使用中", value: stats.using, icon: <LinkOutlined />, color: "#faad14" },
{ key: "formal", label: "正式授权", value: stats.formal, icon: <CheckCircleOutlined />, color: "#52c41a" },
{ key: "available", label: "可分配", value: stats.available, icon: <ClockCircleOutlined />, color: "#13c2c2" },
], [stats]);
const columns: ColumnsType<LicenseVO> = [
{
title: "授权标识",
@ -139,32 +147,10 @@ export default function LicenseManagement() {
<SectionCard
title="授权码管理"
description="查看当前租户授权池"
contentClassName="license-management-page__content"
>
<SummaryStatCards items={statCards} ariaLabel="授权码统计" />
<DataListPanel
leftActions={
<div className="license-summary-chips" aria-label="授权码统计">
<span className="license-summary-chip">
<KeyOutlined aria-hidden="true" />
<span></span>
<strong className="tabular-nums">{stats.total}</strong>
</span>
<span className="license-summary-chip">
<LinkOutlined aria-hidden="true" />
<span>使</span>
<strong className="tabular-nums">{stats.using}</strong>
</span>
<span className="license-summary-chip">
<CheckCircleOutlined aria-hidden="true" />
<span></span>
<strong className="tabular-nums">{stats.formal}</strong>
</span>
<span className="license-summary-chip">
<ClockCircleOutlined aria-hidden="true" />
<span></span>
<strong className="tabular-nums">{stats.available}</strong>
</span>
</div>
}
rightActions={
<Space wrap>
<Input

View File

@ -16,8 +16,91 @@
min-width: 0;
}
.meeting-detail-section-card .section-card__header {
align-items: flex-start;
gap: 16px 20px;
}
.meeting-detail-section-card .section-card__title-wrap {
min-width: 0;
}
.meeting-detail-section-card .section-card__title {
min-width: 0;
padding-bottom: 14px;
align-items: flex-start;
overflow: visible;
white-space: normal;
text-overflow: clip;
}
.meeting-detail-section-card .section-card__title::before {
display: none;
}
.meeting-detail-page-v2 .section-card__extra {
align-self: flex-start;
padding-top: 2px;
}
.meeting-detail-page-v2 .section-card__extra .ant-space {
justify-content: flex-end;
row-gap: 8px;
}
.meeting-detail-page-v2 .section-card__extra .ant-btn {
display: inline-flex;
align-items: center;
justify-content: center;
height: 32px;
}
.meeting-detail-page-v2 .section-card__extra .ant-btn .anticon {
display: inline-flex;
align-items: center;
justify-content: center;
line-height: 1;
}
.meeting-detail-page-v2 .meeting-detail-title-wrap {
display: flex;
align-items: flex-start;
gap: 16px;
min-width: 0;
max-width: 100%;
}
.meeting-detail-page-v2 .meeting-detail-title-icon {
width: 42px;
height: 42px;
flex: 0 0 42px;
display: inline-flex;
align-items: center;
justify-content: center;
line-height: 1;
font-size: 20px;
}
.meeting-detail-page-v2 .meeting-detail-title-icon .anticon {
display: inline-flex;
align-items: center;
justify-content: center;
line-height: 1;
}
.meeting-detail-page-v2 .meeting-detail-title-copy {
min-width: 0;
display: flex;
flex-direction: column;
gap: 7px;
}
.meeting-detail-page-v2 .meeting-detail-title-row {
display: flex;
align-items: center;
gap: 8px;
min-width: 0;
max-width: 100%;
}
.meeting-detail-section-title {
@ -33,6 +116,8 @@
}
.meeting-detail-page-v2 .meeting-detail-title-text {
display: block;
flex: 0 1 auto;
min-width: 0;
overflow: hidden;
color: #333;
@ -60,12 +145,55 @@
.meeting-detail-page-v2 .meeting-detail-meta-row {
display: inline-flex;
flex-wrap: wrap;
gap: 10px 16px;
align-items: center;
gap: 8px 16px;
color: #9095a1;
font-size: 14px;
line-height: 24px;
}
.meeting-detail-page-v2 .meeting-detail-meta-row span {
display: inline-flex;
align-items: center;
gap: 5px;
min-height: 22px;
}
.meeting-detail-page-v2 .meeting-detail-meta-row .anticon {
display: inline-flex;
align-items: center;
justify-content: center;
line-height: 1;
}
.meeting-detail-page-v2 .meeting-detail-config-row {
display: flex;
flex-wrap: wrap;
align-items: center;
gap: 8px 10px;
margin-top: 2px;
}
.meeting-detail-page-v2 .meeting-detail-config-item {
display: inline-flex;
align-items: center;
gap: 8px;
min-height: 34px;
padding: 6px 12px;
border: 1px solid #d8e3ff;
border-radius: 999px;
background: #f3f7ff;
color: #58627f;
font-size: 13px;
line-height: 20px;
white-space: nowrap;
}
.meeting-detail-page-v2 .meeting-detail-config-item strong {
color: #333;
font-weight: 600;
}
.meeting-detail-section-content {
overflow: hidden;
}
@ -422,6 +550,16 @@
}
@media (max-width: 1200px) {
.meeting-detail-section-card .section-card__header {
flex-direction: column;
}
.meeting-detail-page-v2 .section-card__extra,
.meeting-detail-page-v2 .section-card__extra .ant-space {
width: 100%;
justify-content: flex-start;
}
.meeting-detail-section-content {
overflow: auto;
}
@ -438,13 +576,190 @@
}
@media (max-width: 768px) {
.meeting-detail-page-v2 .section-card__extra,
.meeting-detail-page-v2 .section-card__extra .ant-space {
width: 100%;
justify-content: flex-start;
.meeting-detail-page-v2 .meeting-detail-title-wrap {
gap: 12px;
}
.meeting-detail-page-v2 .meeting-detail-title-icon {
width: 38px;
height: 38px;
flex-basis: 38px;
font-size: 18px;
}
.meeting-detail-page-v2 .meeting-detail-title-text {
white-space: normal;
}
.meeting-detail-page-v2 .meeting-detail-title-row {
align-items: flex-start;
}
.meeting-detail-page-v2 .meeting-detail-config-item {
white-space: normal;
}
}
.meeting-detail-page-v2 .detail-right-column {
padding-right: 0;
}
.meeting-detail-page-v2 .transcript-player-anchor > .transcript-workspace-card,
.meeting-detail-page-v2 .transcript-player-anchor > .transcript-workspace-card.left-flow-card,
.meeting-detail-page-v2 .transcript-player-anchor > .transcript-workspace-card.ant-card {
overflow: hidden !important;
border: 1px solid #e5e7eb !important;
border-radius: 18px !important;
background: #ffffff !important;
box-shadow: none !important;
}
.meeting-detail-page-v2 .transcript-panel-header {
display: flex;
align-items: center;
justify-content: flex-start;
flex-shrink: 0;
min-height: 56px;
padding: 0 24px;
border-bottom: 1px solid #e5e7eb;
background: #ffffff;
}
.meeting-detail-page-v2 .transcript-panel-header .transcript-stage-tabs {
display: inline-flex !important;
align-items: center;
align-self: stretch;
gap: 24px !important;
min-height: 56px !important;
padding: 0 !important;
border: 0 !important;
border-radius: 0 !important;
background: transparent !important;
}
.meeting-detail-page-v2 .transcript-panel-header .transcript-stage-tabs button {
position: relative;
height: 56px !important;
min-width: 64px;
padding: 0 !important;
border: 0 !important;
border-radius: 0 !important;
background: transparent !important;
color: #4b5563 !important;
font-size: 16px !important;
font-weight: 700 !important;
box-shadow: none !important;
}
.meeting-detail-page-v2 .transcript-panel-header .transcript-stage-tabs button + button {
margin-left: 0 !important;
}
.meeting-detail-page-v2 .transcript-panel-header .transcript-stage-tabs button:hover {
color: #3c70f5 !important;
background: transparent !important;
}
.meeting-detail-page-v2 .transcript-panel-header .transcript-stage-tabs button.active {
color: #2f5edb !important;
background: transparent !important;
box-shadow: none !important;
}
.meeting-detail-page-v2 .transcript-panel-header .transcript-stage-tabs button.active::after {
content: "" !important;
position: absolute;
right: 0;
bottom: 0;
left: 0;
height: 3px;
border-radius: 999px 999px 0 0;
background: #6258ff;
}
.meeting-detail-page-v2 .transcript-scroll-shell {
padding: 24px 20px 18px 18px !important;
background: #ffffff !important;
}
.meeting-detail-page-v2 .transcript-player.transcript-player--floating {
display: flex;
align-items: center;
gap: 14px;
padding: 14px 16px !important;
border: 0 !important;
border-radius: 18px !important;
background: rgba(255, 255, 255, 0.96) !important;
box-shadow: 0 16px 40px rgba(20, 35, 70, 0.16) !important;
backdrop-filter: blur(12px) !important;
}
.meeting-detail-page-v2 .transcript-player .player-main-btn {
width: 46px !important;
height: 46px !important;
flex: 0 0 46px;
border: 0 !important;
border-radius: 50% !important;
background: linear-gradient(135deg, #3c70f5 0%, #6b8cff 100%) !important;
color: #ffffff !important;
box-shadow: 0 10px 24px rgba(60, 112, 245, 0.32) !important;
}
.meeting-detail-page-v2 .transcript-player .player-ghost-btn {
height: 36px !important;
border: 1px solid #dce6ff !important;
border-radius: 999px !important;
background: #f4f7ff !important;
color: #3c70f5 !important;
box-shadow: none !important;
}
.meeting-detail-page-v2 .transcript-player .player-progress-shell {
flex: 1;
min-width: 0;
}
.meeting-detail-page-v2 .transcript-player .player-range {
height: 6px !important;
border-radius: 999px !important;
background: linear-gradient(
90deg,
#3c70f5 0%,
#3c70f5 var(--player-progress, 0%),
#e6ebf5 var(--player-progress, 0%),
#e6ebf5 100%
) !important;
}
.meeting-detail-page-v2 .transcript-player .player-range::-webkit-slider-thumb {
width: 16px;
height: 16px;
border: 3px solid #ffffff !important;
border-radius: 50% !important;
background: #3c70f5 !important;
box-shadow: 0 4px 10px rgba(60, 112, 245, 0.28) !important;
}
.meeting-detail-page-v2 .transcript-player .player-range::-moz-range-thumb {
width: 16px;
height: 16px;
border: 3px solid #ffffff !important;
border-radius: 50% !important;
background: #3c70f5 !important;
box-shadow: 0 4px 10px rgba(60, 112, 245, 0.28) !important;
}
@media (max-width: 768px) {
.meeting-detail-page-v2 .transcript-panel-header {
align-items: flex-start;
flex-direction: column;
}
.meeting-detail-page-v2 .transcript-panel-header .transcript-stage-tabs {
width: 100%;
}
.meeting-detail-page-v2 .transcript-panel-header .transcript-stage-tabs button {
flex: 1;
}
}

View File

@ -153,6 +153,8 @@ const resolveAudioExtension = (audioUrl?: string) => {
return normalizedUrl.match(/\.([a-z0-9]+)$/i)?.[1]?.toLowerCase() || 'mp3';
};
const normalizeTagDisplay = (tag: string) => tag.replace(/^#+\s*/, '').trim();
const getMeetingAudioDownloadName = (meeting?: Pick<MeetingVO, 'title' | 'audioUrl' | 'playbackAudioUrl'> | null) => {
const audioUrl = resolveMeetingPlaybackAudioUrl(meeting);
const extension = resolveAudioExtension(audioUrl);
@ -2357,6 +2359,10 @@ const MeetingDetail: React.FC = () => {
);
}
const playerProgressPercent = audioDuration > 0
? Math.min(100, Math.max(0, (audioCurrentTime / audioDuration) * 100))
: 0;
return (
<PageContainer title={null} className="meeting-detail-page-v2">
<SectionCard
@ -2380,7 +2386,9 @@ const MeetingDetail: React.FC = () => {
<span>
<ClockCircleOutlined /> {dayjs(meeting.meetingTime).format('YYYY-MM-DD HH:mm')}
</span>
<span>{meeting.participants || '未指定'}</span>
<span>
<UserOutlined /> {meeting.participants || '未指定'}
</span>
</div>
<div className="meeting-detail-config-row">
<span className="meeting-detail-config-item">
@ -2616,7 +2624,7 @@ const MeetingDetail: React.FC = () => {
}
}}
>
<span>#{tag}</span>
<span>{normalizeTagDisplay(tag)}</span>
{isOwner && isSelected && <CheckCircleFilled style={{ fontSize: 12 }} />}
</div>
);
@ -2773,23 +2781,29 @@ const MeetingDetail: React.FC = () => {
</audio>
)}
<div className="transcript-stage-tabs">
{aiCatalogEnabled && (
<div className="transcript-panel-header">
<div className="transcript-stage-tabs" role="tablist" aria-label="内容切换">
{aiCatalogEnabled && (
<button
type="button"
role="tab"
aria-selected={workspaceTab === 'catalog'}
className={workspaceTab === 'catalog' ? 'active' : ''}
onClick={() => setWorkspaceTab('catalog')}
>
AI
</button>
)}
<button
type="button"
className={workspaceTab === 'catalog' ? 'active' : ''}
onClick={() => setWorkspaceTab('catalog')}
role="tab"
aria-selected={workspaceTab === 'transcript'}
className={workspaceTab === 'transcript' ? 'active' : ''}
onClick={() => setWorkspaceTab('transcript')}
>
AI
</button>
)}
<button
type="button"
className={workspaceTab === 'transcript' ? 'active' : ''}
onClick={() => setWorkspaceTab('transcript')}
>
</button>
</div>
</div>
<div className="transcript-scroll-shell">
@ -2920,6 +2934,7 @@ const MeetingDetail: React.FC = () => {
max={audioDuration || 0}
step={0.1}
value={Math.min(audioCurrentTime, audioDuration || 0)}
style={{ '--player-progress': `${playerProgressPercent}%` } as React.CSSProperties}
onChange={handleAudioProgressChange}
/>
</div>
@ -2956,11 +2971,10 @@ const MeetingDetail: React.FC = () => {
}
/* 当转录行处于活动状态时,调整高亮样式以保持可读性 */
.ant-list-item.transcript-row.active .highlight-text {
background: rgba(255, 255, 255, 0.2);
border-bottom-color: #fff;
color: #fff;
text-shadow: 0 1px 2px rgba(0, 0, 0, 0.1);
animation: none; /* 活动行内不需要闪烁,避免视觉混乱 */
background: linear-gradient(120deg, rgba(95, 81, 255, 0.15) 0%, rgba(108, 140, 255, 0.1) 100%);
border-bottom-color: #5f51ff;
color: #4335eb;
text-shadow: none;
}
.summary-keyword-link {
color: #5f51ff;
@ -3905,16 +3919,17 @@ const MeetingDetail: React.FC = () => {
}
.ant-list-item.transcript-row.linked .transcript-bubble,
.ant-list-item.transcript-row.linked .transcript-bubble-editing {
background: linear-gradient(135deg, rgba(95, 81, 255, 0.08), rgba(108, 140, 255, 0.06));
border-color: rgba(95, 81, 255, 0.2);
box-shadow: 0 10px 24px rgba(95, 81, 255, 0.06);
background: #ffffff;
border-color: rgba(60, 112, 245, 0.28);
color: #2d3553;
box-shadow: 0 10px 24px rgba(60, 112, 245, 0.08);
}
.ant-list-item.transcript-row.active .transcript-bubble,
.ant-list-item.transcript-row.active .transcript-bubble-editing {
border-color: rgba(95, 81, 255, 0.16);
background: linear-gradient(135deg, #5b41ff, #6a5cff);
color: #ffffff;
box-shadow: 0 12px 28px rgba(95, 81, 255, 0.2);
border-color: rgba(60, 112, 245, 0.34);
background: #ffffff;
color: #2d3553;
box-shadow: 0 12px 28px rgba(60, 112, 245, 0.12);
}
.transcript-entry {
flex: 1;

View File

@ -92,7 +92,7 @@
min-height: 0;
overflow-x: hidden;
overflow-y: auto;
padding: 4px;
padding: 6px 8px 12px;
}
.meetings-card-scroll .ant-list {
@ -122,29 +122,33 @@
display: flex;
flex-direction: column;
overflow: hidden;
border: 1px solid #e6e6e6 !important;
border: none !important;
border-radius: 16px !important;
background: #fff !important;
box-shadow: 0 2px 8px rgba(24, 39, 75, 0.04) !important;
box-shadow:
0 10px 26px rgba(24, 39, 75, 0.07),
0 2px 8px rgba(24, 39, 75, 0.05) !important;
cursor: pointer;
transform: translateY(0);
will-change: transform, box-shadow;
transition:
border-color 0.18s ease,
background 0.18s ease,
box-shadow 0.18s ease,
transform 0.18s ease;
}
.meeting-card-v2.ant-card:hover {
border-color: #8fb0fb !important;
background: #f9fafe !important;
box-shadow: 0 8px 20px rgba(60, 112, 245, 0.14) !important;
background: linear-gradient(180deg, #ffffff 0%, #fbfdff 100%) !important;
box-shadow:
0 16px 34px rgba(24, 39, 75, 0.12),
0 6px 14px rgba(60, 112, 245, 0.10) !important;
transform: translateY(-3px);
}
.meeting-card-v2.ant-card:active {
box-shadow: 0 4px 12px rgba(60, 112, 245, 0.12) !important;
box-shadow:
0 8px 18px rgba(24, 39, 75, 0.10),
0 2px 8px rgba(60, 112, 245, 0.10) !important;
transform: translateY(-1px);
}

View File

@ -249,11 +249,13 @@ const TableStatusCell: React.FC<{ meeting: MeetingVO; progress: MeetingProgress
};
const getMeetingTagList = (tags: unknown) => {
const normalizeMeetingTag = (tag: unknown) => String(tag).replace(/^#+\s*/, '').trim();
if (Array.isArray(tags)) {
return tags.map((tag) => String(tag).trim()).filter(Boolean);
return tags.map(normalizeMeetingTag).filter(Boolean);
}
if (typeof tags === "string") {
return tags.split(",").map((tag) => tag.trim()).filter(Boolean);
return tags.split(",").map(normalizeMeetingTag).filter(Boolean);
}
return [];
};
@ -382,7 +384,7 @@ const MeetingCardItem: React.FC<{
{tags.length > 0 ? (
tags.map(tag => (
<span key={tag} className="meeting-card-v2__tag">
#{tag}
{tag}
</span>
))
) : (

View File

@ -17,27 +17,6 @@
padding: 8px;
}
.dashboard-monitor-page__stats {
flex-shrink: 0;
min-width: 0;
}
.dashboard-monitor-page__stat-card {
height: 100%;
border: 1px solid #e6e6e6;
border-radius: 4px;
background: #fff;
box-shadow: none;
}
.dashboard-monitor-page__stat-card .ant-card-body {
padding: 18px 24px;
}
.dashboard-monitor-page__stat-label {
font-size: 13px;
}
.dashboard-monitor-page__task-panel {
flex: 1;
min-height: 0;

View File

@ -3,7 +3,8 @@ import PageContainer from "@/components/shared/PageContainer";
import AppPagination from '@/components/shared/AppPagination';
import DataListPanel from "@/components/shared/DataListPanel";
import SectionCard from "@/components/shared/SectionCard";
import { Row, Col, Card, Statistic, List, Tag, Typography, Button, Space, Empty, Steps, Progress, Divider } from 'antd';
import SummaryStatCards from "@/components/shared/SummaryStatCards";
import { Row, Col, List, Tag, Typography, Button, Space, Empty, Steps, Progress, Divider } from 'antd';
import {
HistoryOutlined,
CheckCircleOutlined,
@ -159,15 +160,16 @@ export const Dashboard: React.FC = () => {
};
const statCards = [
{ label: '累计会议记录', value: stats?.totalMeetings, icon: <HistoryOutlined />, color: '#1890ff' },
{ key: 'totalMeetings', label: '累计会议记录', value: stats?.totalMeetings ?? 0, icon: <HistoryOutlined />, color: '#1890ff' },
{
key: 'processingTasks',
label: '当前分析中任务',
value: stats?.processingTasks,
value: stats?.processingTasks ?? 0,
icon: processingCount > 0 ? <LoadingOutlined spin /> : <ClockCircleOutlined />,
color: '#faad14'
},
{ label: '今日新增分析', value: stats?.todayNew, icon: <RiseOutlined />, color: '#52c41a' },
{ label: 'AI 处理成功率', value: `${stats?.successRate || 100}%`, icon: <CheckCircleOutlined />, color: '#13c2c2' },
{ key: 'todayNew', label: '今日新增分析', value: stats?.todayNew ?? 0, icon: <RiseOutlined />, color: '#52c41a' },
{ key: 'successRate', label: 'AI 处理成功率', value: `${stats?.successRate || 100}%`, icon: <CheckCircleOutlined />, color: '#13c2c2' },
];
return (
@ -180,22 +182,7 @@ export const Dashboard: React.FC = () => {
description="系统运行概览与最近任务动态。"
contentClassName="dashboard-monitor-page__content"
>
<div className="dashboard-monitor-page__stats">
<Row gutter={[16, 16]}>
{statCards.map((s, idx) => (
<Col xs={24} sm={12} xl={6} key={idx}>
<Card className="dashboard-monitor-page__stat-card" variant="borderless">
<Statistic
title={<Text type="secondary" className="dashboard-monitor-page__stat-label">{s.label}</Text>}
value={s.value || 0}
valueStyle={{ color: s.color, fontWeight: 700 }}
prefix={React.cloneElement(s.icon as React.ReactElement, { style: { marginRight: 8 } })}
/>
</Card>
</Col>
))}
</Row>
</div>
<SummaryStatCards items={statCards} ariaLabel="任务监控统计" />
<DataListPanel
className="dashboard-monitor-page__task-panel"

View File

@ -12,37 +12,9 @@
background: transparent;
}
.devices-summary-chips {
display: flex;
align-items: center;
gap: 8px;
flex-wrap: wrap;
min-width: 0;
}
.devices-summary-chip {
height: 32px;
display: inline-flex;
align-items: center;
gap: 6px;
padding: 0 10px;
border: 1px solid #e6e6e6;
border-radius: 4px;
background: #f9fafe;
color: #333;
font-size: 13px;
line-height: 20px;
white-space: nowrap;
}
.devices-summary-chip .anticon {
color: #3c70f5;
}
.devices-summary-chip strong {
color: #1677ff;
font-size: 16px;
font-weight: 600;
.devices-page__content {
gap: 12px;
padding: 8px;
}
.device-icon-placeholder {
@ -115,9 +87,4 @@
.devices-page {
padding: 8px;
}
.devices-summary-chips,
.devices-summary-chip {
width: 100%;
}
}

View File

@ -8,6 +8,7 @@ import DataListPanel from "@/components/shared/DataListPanel";
import ListTable from "@/components/shared/ListTable/ListTable";
import AppPagination from "@/components/shared/AppPagination";
import SectionCard from "@/components/shared/SectionCard";
import SummaryStatCards from "@/components/shared/SummaryStatCards";
import { useDict } from "@/hooks/useDict";
import { usePermission } from "@/hooks/usePermission";
import { getDefaultPageSize } from "@/utils/pagination";
@ -93,6 +94,12 @@ export default function Devices() {
return { total, online, enabled };
}, [devices]);
const statCards = useMemo(() => [
{ key: "total", label: t("devicesExt.totalDevices"), value: stats.total, icon: <DesktopOutlined />, color: "#1890ff" },
{ key: "online", label: t("devicesExt.onlineDevices"), value: stats.online, icon: <ThunderboltOutlined />, color: "#faad14" },
{ key: "enabled", label: t("devicesExt.enabledDevices"), value: stats.enabled, icon: <CheckCircleOutlined />, color: "#1677ff" },
], [stats, t]);
const openEdit = (record: DeviceInfo) => {
setEditing(record);
form.setFieldsValue({
@ -305,27 +312,10 @@ export default function Devices() {
<SectionCard
title={t("devices.title")}
description={t("devices.subtitle")}
contentClassName="devices-page__content"
>
<SummaryStatCards items={statCards} ariaLabel={t("devices.title")} />
<DataListPanel
leftActions={
<div className="devices-summary-chips" aria-label={t("devices.title")}>
<span className="devices-summary-chip">
<DesktopOutlined aria-hidden="true" />
<span>{t("devicesExt.totalDevices")}</span>
<strong className="tabular-nums">{stats.total}</strong>
</span>
<span className="devices-summary-chip">
<ThunderboltOutlined aria-hidden="true" />
<span>{t("devicesExt.onlineDevices")}</span>
<strong className="tabular-nums">{stats.online}</strong>
</span>
<span className="devices-summary-chip">
<CheckCircleOutlined aria-hidden="true" />
<span>{t("devicesExt.enabledDevices")}</span>
<strong className="tabular-nums">{stats.enabled}</strong>
</span>
</div>
}
rightActions={
<Space wrap>
<Input

View File

@ -28,7 +28,7 @@
display: flex;
flex-direction: column;
overflow-y: auto;
padding: 8px 4px 0;
padding: 8px 8px 12px;
}
.tenants-grid {
@ -53,29 +53,33 @@
min-height: 230px;
display: flex;
flex-direction: column;
border: 1px solid #e6e6e6 !important;
border: none !important;
border-radius: 16px !important;
background: #fff !important;
box-shadow: 0 2px 8px rgba(24, 39, 75, 0.04) !important;
box-shadow:
0 10px 26px rgba(24, 39, 75, 0.07),
0 2px 8px rgba(24, 39, 75, 0.05) !important;
overflow: hidden;
transform: translateY(0);
will-change: transform, box-shadow;
transition:
border-color 0.18s ease,
background 0.18s ease,
box-shadow 0.18s ease,
transform 0.18s ease;
}
.tenant-card:hover {
border-color: #8fb0fb !important;
background: #f9fafe !important;
box-shadow: 0 8px 20px rgba(60, 112, 245, 0.14) !important;
background: linear-gradient(180deg, #ffffff 0%, #fbfdff 100%) !important;
box-shadow:
0 16px 34px rgba(24, 39, 75, 0.12),
0 6px 14px rgba(60, 112, 245, 0.10) !important;
transform: translateY(-3px);
}
.tenant-card:active {
box-shadow: 0 4px 12px rgba(60, 112, 245, 0.12) !important;
box-shadow:
0 8px 18px rgba(24, 39, 75, 0.10),
0 2px 8px rgba(60, 112, 245, 0.10) !important;
transform: translateY(-1px);
}

View File

@ -52,6 +52,18 @@
min-height: 0; /* Important for flex child scroll */
}
.dict-type-search-row {
display: flex;
gap: 8px;
width: 100%;
margin-bottom: 12px;
}
.dict-type-search-input {
flex: 1;
min-width: 0;
}
.full-height {
height: 100%;
}

View File

@ -183,8 +183,8 @@ export default function Dictionaries() {
}} extra={can("sys_dict:type:create") &&
<Button type="primary" size="small" icon={<PlusOutlined aria-hidden="true"/>}
onClick={handleAddType}>{t("common.create")}</Button>}>
<div style={{ marginBottom: 12 }} className="flex-shrink-0">
<Space.Compact style={{ width: "100%" }}>
<div className="dict-type-search-row flex-shrink-0">
<Space.Compact className="dict-type-search-input">
<Input
placeholder={t("dictsExt.searchTypes")}
allowClear
@ -193,8 +193,8 @@ export default function Dictionaries() {
onPressEnter={() => handleTypeSearch(typeKeyword)}
/>
<Button type="primary" icon={<SearchOutlined aria-hidden="true" />} onClick={() => handleTypeSearch(typeKeyword)}>{t("common.search")}</Button>
<Button onClick={handleTypeReset}>{t("common.reset")}</Button>
</Space.Compact>
<Button onClick={handleTypeReset}>{t("common.reset")}</Button>
</div>
<div className="dict-type-table-wrap">
<Table