106 lines
4.2 KiB
TypeScript
106 lines
4.2 KiB
TypeScript
import { Search } from 'lucide-react';
|
||
import React, { useState } from 'react';
|
||
|
||
interface MenuItem {
|
||
label: string;
|
||
key: string;
|
||
}
|
||
|
||
// 定义 TopNavProps 接口,描述组件的 props 类型
|
||
// menuItems? 菜单项数组
|
||
// activeKey: 当前激活的菜单项
|
||
// onSearch: 搜索回调函数
|
||
// onItemClick: 菜单点击回调函数
|
||
interface TopNavProps {
|
||
menuItems?: MenuItem[];
|
||
activeKey?: string;
|
||
onSearch?: (keyword: string) => void;
|
||
onItemClick?: (key: string) => void;
|
||
}
|
||
|
||
//定义 TopNav 组件,类型为 React 函数组件,接收 TopNavProps 类型的 props
|
||
//解构并设置 menuItems 默认值,如果父组件没有传入则使用默认的6个菜单项
|
||
// activeKey: 当前激活的菜单项
|
||
// onSearch: 搜索回调函数
|
||
// onItemClick: 菜单点击回调函数
|
||
export function TopNav({
|
||
menuItems = [
|
||
{ label: '首页', key: 'home' },
|
||
{ label: '烽火动态', key: 'news' },
|
||
{ label: '烽火铸魂', key: 'soul' },
|
||
{ label: '烽火训练', key: 'training' },
|
||
{ label: '联系热线', key: 'hotline' },
|
||
{ label: '综合服务', key: 'service' },
|
||
],
|
||
activeKey: externalActiveKey, // 从外部传入的 activeKey
|
||
onSearch,
|
||
onItemClick,
|
||
}: TopNavProps){
|
||
// 使用外部传入的 activeKey,如果没有则使用内部状态
|
||
// 创建内部状态 internalActiveKey,默认值为 'home',用于内部管理激活状态
|
||
const [internalActiveKey, setInternalActiveKey] = useState('home');
|
||
// 如果外部传入了 activeKey 则使用外部的,否则使用内部状态(支持受控和非受控模式)
|
||
const currentActiveKey = externalActiveKey !== undefined ? externalActiveKey : internalActiveKey;
|
||
// 创建搜索关键词状态,初始为空字符串
|
||
const [searchKeyword, setSearchKeyword] = useState('');
|
||
// 处理搜索提交事件, 阻止默认表单提交行为,调用搜索回调函数
|
||
const handleSearchSubmit = (e: React.FormEvent) => {
|
||
e.preventDefault();
|
||
onSearch?.(searchKeyword); // .? 可选链操作符,确保 onSearch 存在时才调用
|
||
};
|
||
|
||
// 定义菜单项点击处理函数,如果外部没有传入 activeKey 则更新内部状态,调用点击回调函数
|
||
const handleItemClick = (item: MenuItem) => {
|
||
// 更新内部状态(如果使用内部状态)
|
||
if (externalActiveKey === undefined) {
|
||
setInternalActiveKey(item.key);
|
||
}
|
||
// 调用外部回调函数
|
||
onItemClick?.(item.key); // .? 可选链操作符,确保 onItemClick 存在时才调用
|
||
};
|
||
|
||
return (
|
||
<div className="h-14 flex items-center justify-center px-8 bg-white">
|
||
{/* 搜索框与导航菜单组合 */}
|
||
<div className="flex items-center space-x-4">
|
||
{/* 搜索框 */}
|
||
<form onSubmit={handleSearchSubmit} className="relative">
|
||
<div className="relative">
|
||
<input
|
||
type="text"
|
||
value={searchKeyword}
|
||
onChange={(e) => setSearchKeyword(e.target.value)}
|
||
placeholder="搜索..."
|
||
className="pl-10 pr-4 py-2 text-sm rounded-lg border border-gray-300 focus:outline-none focus:ring-2 focus:ring-blue-500 focus:border-transparent w-64 transition-all duration-200 hover:shadow-sm"
|
||
/>
|
||
<span className="absolute left-3 top-1/2 transform -translate-y-1/2 text-gray-400">
|
||
<Search className="w-5 h-5" />
|
||
</span>
|
||
</div>
|
||
</form>
|
||
|
||
{/* 导航菜单 */}
|
||
<ul className="flex space-x-2">
|
||
{menuItems.map((item) => {
|
||
const isActive = currentActiveKey === item.key; // 判断当前项是否激活
|
||
return (
|
||
<li key={item.key}>
|
||
<button
|
||
onClick={() => handleItemClick(item)}
|
||
className={`px-4 py-2 text-sm font-medium rounded-lg transition-all duration-200 ${
|
||
isActive
|
||
? 'bg-blue-600 text-white shadow-md' // 激活状态样式
|
||
: 'text-gray-600 hover:bg-blue-100 hover:text-blue-700' // 非激活状态样式
|
||
}`}
|
||
>
|
||
{item.label}
|
||
</button>
|
||
</li>
|
||
);
|
||
})}
|
||
</ul>
|
||
</div>
|
||
</div>
|
||
);
|
||
};
|