mirror of
https://github.com/woodchen-ink/nezha-dash-v1.git
synced 2025-07-18 09:31:55 +08:00
185 lines
6.6 KiB
TypeScript
185 lines
6.6 KiB
TypeScript
import useTooltip from "@/hooks/use-tooltip"
|
|
import { geoJsonString } from "@/lib/geo-json-string"
|
|
import { countryCoordinates } from "@/lib/geo-limit"
|
|
import { cn, formatNezhaInfo } from "@/lib/utils"
|
|
import { NezhaServer } from "@/types/nezha-api"
|
|
import { geoEquirectangular, geoPath } from "d3-geo"
|
|
import { useTranslation } from "react-i18next"
|
|
|
|
import MapTooltip from "./MapTooltip"
|
|
|
|
export default function GlobalMap({ serverList, now }: { serverList: NezhaServer[]; now: number }) {
|
|
const { t } = useTranslation()
|
|
const countryList: string[] = []
|
|
const serverCounts: { [key: string]: number } = {}
|
|
|
|
const customBackgroundImage = (window.CustomBackgroundImage as string) !== "" ? window.CustomBackgroundImage : undefined
|
|
|
|
serverList.forEach((server) => {
|
|
if (server.country_code) {
|
|
const countryCode = server.country_code.toUpperCase()
|
|
if (!countryList.includes(countryCode)) {
|
|
countryList.push(countryCode)
|
|
}
|
|
serverCounts[countryCode] = (serverCounts[countryCode] || 0) + 1
|
|
}
|
|
})
|
|
|
|
const width = 900
|
|
const height = 500
|
|
|
|
const geoJson = JSON.parse(geoJsonString)
|
|
const filteredFeatures = geoJson.features.filter((feature: { properties: { iso_a3_eh: string } }) => feature.properties.iso_a3_eh !== "")
|
|
|
|
return (
|
|
<section
|
|
className={cn("flex flex-col gap-4 mt-8", {
|
|
"bg-card/70 rounded-lg p-4": customBackgroundImage,
|
|
})}
|
|
>
|
|
<p className="text-sm font-medium opacity-40">
|
|
{t("map.Distributions")} {countryList.length} {t("map.Regions")}
|
|
</p>
|
|
<div className="w-full overflow-x-auto">
|
|
<InteractiveMap
|
|
countries={countryList}
|
|
serverCounts={serverCounts}
|
|
width={width}
|
|
height={height}
|
|
filteredFeatures={filteredFeatures}
|
|
nezhaServerList={serverList}
|
|
now={now}
|
|
/>
|
|
</div>
|
|
</section>
|
|
)
|
|
}
|
|
|
|
interface InteractiveMapProps {
|
|
countries: string[]
|
|
serverCounts: { [key: string]: number }
|
|
width: number
|
|
height: number
|
|
filteredFeatures: {
|
|
type: "Feature"
|
|
properties: {
|
|
iso_a2_eh: string
|
|
[key: string]: string
|
|
}
|
|
geometry: never
|
|
}[]
|
|
nezhaServerList: NezhaServer[]
|
|
now: number
|
|
}
|
|
|
|
export function InteractiveMap({ countries, serverCounts, width, height, filteredFeatures, nezhaServerList, now }: InteractiveMapProps) {
|
|
const { setTooltipData } = useTooltip()
|
|
|
|
const projection = geoEquirectangular()
|
|
.scale(140)
|
|
.translate([width / 2, height / 2])
|
|
.rotate([-12, 0, 0])
|
|
|
|
const path = geoPath().projection(projection)
|
|
|
|
return (
|
|
<div className="relative w-full aspect-[2/1]" onMouseLeave={() => setTooltipData(null)}>
|
|
<svg width={width} height={height} viewBox={`0 0 ${width} ${height}`} xmlns="http://www.w3.org/2000/svg" className="w-full h-auto">
|
|
<defs>
|
|
<pattern id="dots" width="2" height="2" patternUnits="userSpaceOnUse">
|
|
<circle cx="1" cy="1" r="0.5" fill="currentColor" />
|
|
</pattern>
|
|
</defs>
|
|
<g>
|
|
{/* Background rect to handle mouse events in empty areas */}
|
|
<rect x="0" y="0" width={width} height={height} fill="transparent" onMouseEnter={() => setTooltipData(null)} />
|
|
{filteredFeatures.map((feature, index) => {
|
|
const isHighlighted = countries.includes(feature.properties.iso_a2_eh)
|
|
|
|
const serverCount = serverCounts[feature.properties.iso_a2_eh] || 0
|
|
|
|
return (
|
|
<path
|
|
key={index}
|
|
d={path(feature) || ""}
|
|
className={
|
|
isHighlighted
|
|
? "fill-green-700 hover:fill-green-600 dark:fill-green-900 dark:hover:fill-green-700 transition-all cursor-pointer"
|
|
: "fill-neutral-200/50 dark:fill-neutral-800 stroke-neutral-300/40 dark:stroke-neutral-700 stroke-[0.5]"
|
|
}
|
|
onMouseEnter={() => {
|
|
if (!isHighlighted) {
|
|
setTooltipData(null)
|
|
return
|
|
}
|
|
if (path.centroid(feature)) {
|
|
const countryCode = feature.properties.iso_a2_eh
|
|
const countryServers = nezhaServerList
|
|
.filter((server: NezhaServer) => server.country_code?.toUpperCase() === countryCode)
|
|
.map((server: NezhaServer) => ({
|
|
name: server.name,
|
|
status: formatNezhaInfo(now, server).online,
|
|
}))
|
|
setTooltipData({
|
|
centroid: path.centroid(feature),
|
|
country: feature.properties.name,
|
|
count: serverCount,
|
|
servers: countryServers,
|
|
})
|
|
}
|
|
}}
|
|
/>
|
|
)
|
|
})}
|
|
|
|
{/* 渲染不在 filteredFeatures 中的国家标记点 */}
|
|
{countries.map((countryCode) => {
|
|
// 检查该国家是否已经在 filteredFeatures 中
|
|
const isInFilteredFeatures = filteredFeatures.some((feature) => feature.properties.iso_a2_eh === countryCode)
|
|
|
|
// 如果已经在 filteredFeatures 中,跳过
|
|
if (isInFilteredFeatures) return null
|
|
|
|
// 获取国家的经纬度
|
|
const coords = countryCoordinates[countryCode]
|
|
if (!coords) return null
|
|
|
|
// 使用投影函数将经纬度转换为 SVG 坐标
|
|
const [x, y] = projection([coords.lng, coords.lat]) || [0, 0]
|
|
const serverCount = serverCounts[countryCode] || 0
|
|
|
|
return (
|
|
<g
|
|
key={countryCode}
|
|
onMouseEnter={() => {
|
|
const countryServers = nezhaServerList
|
|
.filter((server: NezhaServer) => server.country_code?.toUpperCase() === countryCode.toUpperCase())
|
|
.map((server: NezhaServer) => ({
|
|
name: server.name,
|
|
status: formatNezhaInfo(now, server).online,
|
|
}))
|
|
setTooltipData({
|
|
centroid: [x, y],
|
|
country: coords.name,
|
|
count: serverCount,
|
|
servers: countryServers,
|
|
})
|
|
}}
|
|
className="cursor-pointer"
|
|
>
|
|
<circle
|
|
cx={x}
|
|
cy={y}
|
|
r={4}
|
|
className="fill-sky-700 stroke-white hover:fill-sky-600 dark:fill-sky-900 dark:hover:fill-sky-700 transition-all"
|
|
/>
|
|
</g>
|
|
)
|
|
})}
|
|
</g>
|
|
</svg>
|
|
<MapTooltip />
|
|
</div>
|
|
)
|
|
}
|