From a57caa72a54efe9de3fe26a6c36d3e8038267377 Mon Sep 17 00:00:00 2001
From: guonan <guonan201020@163.com>
Date: 星期四, 17 七月 2025 09:09:49 +0800
Subject: [PATCH] 修改x按钮

---
 src/components/menu/TimeLine.vue | 1382 +++++++++++++++++++++++++++++++++++++++++++++++++++++++++
 1 files changed, 1,382 insertions(+), 0 deletions(-)

diff --git a/src/components/menu/TimeLine.vue b/src/components/menu/TimeLine.vue
new file mode 100644
index 0000000..6951358
--- /dev/null
+++ b/src/components/menu/TimeLine.vue
@@ -0,0 +1,1382 @@
+<template>
+  <div class="timeline-container">
+    <div class="controls">
+      <div class="control-btn" @click="skipBackward">
+        <img
+          src="@/assets/img/timeline/left.png"
+          class="fas fa-step-backward"
+        />
+      </div>
+      <div class="control-btn play-btn" @click="togglePlay">
+        <img v-show="isPlaying" src="@/assets/img/timeline/stop.png" />
+        <img v-show="!isPlaying" src="@/assets/img/timeline/start.png" />
+      </div>
+      <div class="control-btn" @click="skipForward">
+        <img
+          src="@/assets/img/timeline/right.png"
+          class="fas fa-step-forward"
+        />
+      </div>
+      <div class="speed-control" v-show="speedShow">
+        <div @click="toggleSpeedMenu">{{ playbackRate }}X</div>
+        <div class="speed-menu" v-show="showSpeedMenu">
+          <div
+            v-for="rate in playbackRates"
+            :key="rate"
+            @click.capture="setPlaybackRate(rate)"
+            :class="{ active: playbackRate === rate }"
+          >
+            {{ rate }}X
+          </div>
+        </div>
+      </div>
+    </div>
+
+    <div class="timeline">
+      <div class="dates">
+        <div class="current-date">褰撳墠鎾斁鏃堕棿锛歿{ currentPlayingTime }}</div>
+        <div
+          v-for="(date, index) in visibleDates"
+          :key="index"
+          class="date-label"
+        >
+          <!-- {{ formatDate(date) }} -->
+        </div>
+        <div>
+          涓撻娓叉煋:
+          <el-switch
+            v-model="isColorRenderEnabled"
+            @change="handleColorRenderChange"
+            style="margin-top: -3px"
+            :disabled="!isPlaying || !isWaterPrimitiveCreated"
+          />
+          <!-- active-text="寮�" inactive-text="鍏�" -->
+        </div>
+      </div>
+      <div class="timeline-track" ref="timelineTrack" @click="seekToPosition">
+        <div
+          class="timeline-progress"
+          :style="{ width: progressPercentage + '%' }"
+        ></div>
+        <div
+          class="timeline-cursor"
+          :style="{ left: progressPercentage + '%' }"
+        ></div>
+        <div class="scale-markers">
+          <div class="scale-marker" style="left: 0%"></div>
+          <div class="scale-marker" style="left: 25%"></div>
+          <div class="scale-marker" style="left: 50%"></div>
+          <div class="scale-marker" style="left: 75%"></div>
+          <div class="scale-marker" style="left: 100%"></div>
+        </div>
+        <div class="time-markers">
+          <div
+            v-for="(time, index) in timeMarkers"
+            :key="index"
+            class="time-marker"
+            :style="{ left: `${index * 25}%`, transform: 'translateX(-50%)' }"
+          >
+            <div class="date-part">{{ time.split(" ")[0] }}</div>
+            <div class="time-part">{{ time.split(" ")[1] }}</div>
+          </div>
+        </div>
+      </div>
+    </div>
+    <div>
+      <div style="display: flex">
+        <ratelevel
+          ref="ratelevelRef"
+          :playing-time="sendCurrentPlayingTime"
+          @finish-calculation="handleFinishCalculation"
+          style="
+            margin-top: 12px;
+            margin-left: 28px;
+            margin-right: 10px;
+            justify-content: flex-end;
+          "
+        />
+        <crossanalysis
+          ref="crossRef"
+          style="
+            margin-top: 12px;
+            margin-left: 16px;
+            margin-right: 20px;
+            justify-content: flex-end;
+          "
+        />
+      </div>
+      <el-button
+        @click="handleBack"
+        style="
+          margin-top: 3px;
+          margin-left: 28px;
+          margin-right: 10px;
+          width: 75%;
+          height: 30%;
+        "
+        >缁撴潫妯℃嫙</el-button
+      >
+    </div>
+  </div>
+</template>
+
+<script setup>
+import {
+  ref,
+  computed,
+  onMounted,
+  watch,
+  defineProps,
+  onBeforeUnmount,
+  inject,
+  reactive,
+  watchEffect,
+} from "vue";
+import ratelevel from "@/components/menu/flowRate_waterLevel.vue";
+import crossanalysis from "@/components/menu/CrossSectionalAnalysis.vue";
+
+import dayjs from "dayjs";
+import {
+  createWaterPrimitive,
+  destoryWaterPrimitive,
+  pauseWaterSimulation,
+  resumeWaterSimulation,
+  setTimeForWaterSimulation,
+  toggleWaterColorRender,
+  updateWaterColor,
+} from "@/utils/water";
+import mapUtils from "@/utils/tools.js";
+import { fetchWaterSimulationData, stopSim } from "@/api/trApi.js";
+import { EventBus } from "@/eventBus";
+import { ElMessage, ElMessageBox } from "element-plus";
+// 鐘舵�佺鐞嗗櫒
+import { useSimStore } from "@/store/simulation";
+import { storeToRefs } from "pinia";
+const simStore = useSimStore();
+const { selectedScheme, frameNum, layerDate, schemWaterInfo } =
+  storeToRefs(simStore);
+import { clearAllPoints } from "@/utils/map";
+
+const emit = defineEmits([
+  "timeUpdate",
+  "isPlaying",
+  "playbackFinished",
+  "isColorRender",
+]);
+// 瀹氫箟props
+const props = defineProps({
+  waterSimulateParams: {
+    type: Object,
+    default: () => ({
+      date: ["2025-02-14T16:00:00.000Z", "2025-02-15T16:00:00.000Z"],
+    }),
+  },
+});
+// 鍝嶅簲寮忕姸鎬�
+let serviceInfo = ref(null); // 褰撳墠鏂规鐨勬湇鍔″湴鍧�
+const ratelevelRef = ref(null); // 鑾峰彇瀛愮粍浠跺疄渚嬬殑寮曠敤
+const crossRef = ref(null); // 鑾峰彇瀛愮粍浠跺疄渚嬬殑寮曠敤
+const currentPlayingTime = ref(""); // 褰撳墠鎾斁鏃堕棿
+const sendCurrentPlayingTime = ref(""); // 褰撳墠鎾斁鏃堕棿
+const isPlaying = ref(false);
+const playbackFinished = ref(true);
+const currentTime = ref(0);
+const duration = ref(60); // 涓�澶╃殑绉掓暟
+const playbackRate = ref(1);
+const playbackRates = ref([1, 2, 4, 8]);
+const showSpeedMenu = ref(false);
+const speedShow = ref(false);
+
+const waterTimestamps = ref([]); // 瀛樺偍鏃堕棿杞存暟鎹�
+const timeMarkers = ref([]);
+const timelineTrack = ref(null);
+const isColorRenderEnabled = ref(false); // 鍋囪杩欐槸浣犵殑棰滆壊娓叉煋寮�鍏崇姸鎬�
+const isWaterPrimitiveCreated = ref(false);
+let playInterval = null;
+let timeStepInfo = null;
+let rainTotalInfo = [];
+const isRainEnabled = ref(false);
+const rainParams = reactive({
+  rainSize: 0.5,
+  rainSpeed: 50,
+  rainColor: "#99B3CC",
+  rainDensity: 30, // 闆ㄧ殑瀵嗗害
+});
+let minFlowRate = ref();
+let maxFlowRate = ref();
+// 璁$畻灞炴��
+const progressPercentage = computed(
+  () => (currentTime.value / duration.value) * 100
+);
+const visibleDates = computed(() =>
+  Array.from(
+    new Set(waterTimestamps.value.map((ts) => dayjs(ts).format("YYYY-MM-DD")))
+  ).map((date) => dayjs(date).toDate())
+);
+
+// 鎴戦渶瑕佸姞涓�涓垽鏂�
+const finishPlay = ref(false);
+// 鎾斁鎺у埗
+const togglePlay = () => {
+  // 杩欓噷搴旇鍐嶈瀹氬嚑涓檺鍒讹紝濡傛灉缂哄皯浠�涔堟暟鎹紝鏃犳硶杩涜浠跨湡
+  if (!isPlaying.value && currentTime.value >= duration.value)
+    currentTime.value = 0;
+
+  isPlaying.value = !isPlaying.value;
+  emit("isPlaying", isPlaying.value);
+
+  if (isPlaying.value) {
+    startPlayback();
+
+    if (!isWaterPrimitiveCreated.value) {
+      // console.log(serviceInfo, '杩欓噷鏄綋鍓嶆柟妗堢殑鏈嶅姟淇℃伅锛�');
+      // 杩欓噷閫氳繃water.js涓幓鍙戦�佽姹傝幏鍙栨按闈㈡ā鎷�
+      createWaterPrimitive({
+        baseUrl: `/simu/${serviceInfo}`,
+        // baseUrl: `/simu/c2h1dc`,
+        interval: intervalMap[playbackRate.value],
+        colorRender: isColorRenderEnabled.value,
+        minFlowRate,
+        maxFlowRate,
+      });
+      isWaterPrimitiveCreated.value = true;
+    } else {
+      resumeWaterSimulation();
+      toggleWaterColorRender(isColorRenderEnabled.value); // 鏇存柊棰滆壊娓叉煋
+    }
+
+    if (currentTime.value === 0) emit("playbackFinished", false);
+
+    if (isRainEnabled.value) {
+      mapUtils.toggleRain(rainParams, true);
+    }
+  } else {
+    stopPlayback();
+    pauseWaterSimulation();
+
+    isRainEnabled.value = true;
+    setTimeout(() => {
+      mapUtils.delRain();
+    }, 3000);
+  }
+};
+
+// 棰滆壊娓叉煋鍒囨崲浜嬩欢
+const handleColorRenderChange = (enabled) => {
+  if (!isPlaying.value) {
+    ElMessage({
+      message: "璇峰厛鍚姩姘翠綋妯℃嫙鍚庡啀杩涜涓撻鏁堟灉鍒囨崲銆�",
+      type: "warning",
+    });
+    return; // 闃绘鍚庣画閫昏緫鎵ц
+  }
+  if (isWaterPrimitiveCreated.value) {
+    console.log("褰撳墠鏄惁寮�鍚笓棰樻覆鏌擄細", enabled);
+    emit("isColorRender", enabled);
+    toggleWaterColorRender(enabled);
+  }
+};
+const intervalMap = {
+  1: 1000, // 1鍊嶉��
+  2: 500, // 2鍊嶉��
+  4: 250, // 4鍊嶉��
+  8: 125, // 8鍊嶉��
+};
+// 鎾斁閫昏緫
+// const startPlayback = () => {
+//   clearInterval(playInterval);
+
+//   playInterval = setInterval(() => {
+//     // 鎵惧埌褰撳墠鏃堕棿瀵瑰簲鐨勭储寮�
+//     const currentIndex = findClosestTimestampIndex(currentTime.value);
+//     const nextIndex = currentIndex + 1;
+
+//     // 濡傛灉宸茬粡鏄渶鍚庝竴涓椂闂寸偣锛屽仠姝㈡挱鏀�
+//     if (nextIndex >= waterTimestamps.value.length) {
+//       currentTime.value = duration.value;
+//       stopPlayback();
+//       isPlaying.value = false;
+//       emit("isPlaying", false);
+//       emit("playbackFinished", true);
+//       return;
+//     }
+
+//     // 鏇存柊鏃堕棿涓轰笅涓�涓椂闂寸偣鐨勬椂闂村樊锛堢锛�
+//     const nextTimestamp = waterTimestamps.value[nextIndex];
+//     const baseTimestamp = waterTimestamps.value[0];
+//     currentTime.value = (nextTimestamp - baseTimestamp) / 1000;
+
+//     // 瑙﹀彂鏇存柊
+//     if (selectedScheme.value.type !== 2) {
+//       updateWaterColorByTime();
+//       updateWeatherByProgress();
+//     }
+
+//     const progress = currentTime.value / duration.value;
+//     emit("timeUpdate", progress * 100);
+//   }, 1000 / playbackRate.value); // 鏍规嵁鎾斁閫熺巼璋冩暣闂撮殧
+// };
+const startPlayback = () => {
+  clearInterval(playInterval);
+
+  // 鏂板缓鏂规涓殑瀹炴椂妯℃嫙涓嶈兘鍊嶉��
+  if (selectedScheme.value.type === 2 && simStore.rePlayList.length == 0) {
+    console.log("鏂板缓鏂规瀹炴椂妯℃嫙浜旂涓�璺�");
+    // 瀹炴椂妯℃嫙锛氭瘡 5 绉掕烦鍔ㄤ竴娆�
+    playInterval = setInterval(() => {
+      const fiveSeconds = 5;
+      const totalDuration = duration.value; // 鎬绘椂闀匡紙绉掞級
+
+      currentTime.value += fiveSeconds;
+
+      if (currentTime.value >= totalDuration) {
+        currentTime.value = totalDuration;
+        stopPlayback();
+        isPlaying.value = false;
+        finishPlay.value = true;
+        emit("isPlaying", false);
+        emit("playbackFinished", true);
+        return;
+      }
+
+      // 瑙﹀彂杩涘害鏇存柊
+      // const progress = currentTime.value / totalDuration;
+      // 瀹炴椂妯℃嫙搴旇涓嶇敤鏄剧ず寮圭獥鍚�
+      // emit("timeUpdate", progress * 100);
+
+      // 濡傛灉闇�瑕佽Е鍙戞煇浜涙洿鏂板嚱鏁帮紝涔熷彲浠ヤ繚鐣�
+      updateWaterColorByTime();
+      // updateWeatherByProgress();
+      // 淇敼涓哄浐瀹氶樁娈碉紝缂撴參涓嬮洦鐨勭姸鎬�
+      const rainParams = {
+        rainSize: 0.5,
+        rainSpeed: 20,
+        rainDensity: 15,
+        rainColor: "#ADD8E6",
+      };
+      console.log("瀹炴椂妯℃嫙寮�濮嬩笅闆�");
+      // 璋冪敤宸ュ叿鏂规硶鏇存柊闆ㄦ晥
+      mapUtils.toggleRain(rainParams, true);
+    }, 5000); // 姣� 5 绉掓墽琛屼竴娆�
+  } else {
+    // 杩欓噷闈㈣繕鏄綘鐨勬挱鏀句唬鐮侊紝涓婇潰鐨刬f涓槸浜旂閽熻烦鍔ㄤ竴娆$殑瀹炴椂妯℃嫙
+    playInterval = setInterval(() => {
+      const currentIndex = findClosestTimestampIndex(currentTime.value);
+      const nextIndex = currentIndex + 1;
+
+      if (nextIndex >= waterTimestamps.value.length) {
+        currentTime.value = duration.value;
+        stopPlayback();
+        isPlaying.value = false;
+        finishPlay.value = true;
+        emit("isPlaying", false);
+        emit("playbackFinished", true);
+        return;
+      }
+
+      const nextTimestamp = waterTimestamps.value[nextIndex];
+      const baseTimestamp = waterTimestamps.value[0];
+      currentTime.value = (nextTimestamp - baseTimestamp) / 1000;
+
+      if (selectedScheme.value.type !== 2) {
+        updateWaterColorByTime();
+        updateWeatherByProgress();
+      }
+
+      const progress = currentTime.value / duration.value;
+      if (selectedScheme.value.type !== 2) {
+        emit("timeUpdate", progress * 100);
+      }
+    }, 1000 / playbackRate.value);
+  }
+};
+// 闄嶉洦鍙樺寲閮ㄥ垎
+// 闄嶉洦鏁版嵁鐩稿叧鍙橀噺
+let rainFallValues = ref([]); // 瀛樺偍鍘熷闄嶉洦閲忔暟鎹�
+let minRainValue = ref(Infinity);
+let averageRainIntensity = ref();
+let maxRainValue = ref(-Infinity);
+// 鑾峰彇闄嶉洦鏁版嵁
+function getRainfallData() {
+  if (!selectedScheme.value || !selectedScheme.value.data) {
+    console.warn("selectedScheme 鎴� data 涓嶅瓨鍦�");
+    return;
+  }
+
+  let data = selectedScheme.value.data;
+
+  // 濡傛灉鏄瓧绗︿覆锛屽垯灏濊瘯瑙f瀽涓哄璞�
+  if (typeof data === "string") {
+    try {
+      data = JSON.parse(data);
+      console.log("瑙f瀽鍚庣殑闄嶉洦鏁版嵁锛�", data);
+    } catch (e) {
+      console.error("data 涓嶆槸鏈夋晥鐨� JSON 瀛楃涓�");
+      return;
+    }
+  }
+
+  console.log("闄嶉洦寮哄害鐨勫崟浣嶆槸锛�", data.intensityUnit);
+
+  // 鍒ゆ柇 rainfalls 鏄惁涓哄璞★紝濡傛灉鏄垯杞垚鏁扮粍
+  let rainfalls = data.rainfalls;
+  if (typeof rainfalls === "object" && !Array.isArray(rainfalls)) {
+    rainfalls = Object.values(rainfalls);
+    console.warn("鈿狅笍 rainfalls 鏄璞★紝宸茶浆鎹负鏁扮粍");
+  }
+
+  // 鎸夊皬鏃惰仛鍚堥檷闆ㄦ暟鎹�
+  const hourlyRainfallMap = {};
+
+  rainfalls.forEach((record) => {
+    const originalTime = new Date(record.time);
+    if (isNaN(originalTime.getTime())) {
+      console.warn("鏃犳晥鐨勬椂闂存牸寮忥細", record.time);
+      return;
+    }
+
+    // 鏋勯�犫�滃皬鏃垛�濈骇鍒殑鏃堕棿閿紝姣斿锛�2024-08-25 20:00:00
+    const hourKey = new Date(
+      originalTime.getFullYear(),
+      originalTime.getMonth(),
+      originalTime.getDate(),
+      originalTime.getHours()
+    );
+
+    const hourStr = hourKey.toISOString().slice(0, 16).replace("T", " ");
+
+    if (!hourlyRainfallMap[hourStr]) {
+      hourlyRainfallMap[hourStr] = {
+        intensity: 0,
+        time: hourStr,
+        total: record.total, // 榛樿鐢ㄧ涓�涓褰曠殑 total
+      };
+    }
+
+    hourlyRainfallMap[hourStr].intensity += record.intensity;
+    // 鍙栨渶澶х殑 total锛堝洜涓烘槸绱Н鍊硷級
+    if (record.total > hourlyRainfallMap[hourStr].total) {
+      hourlyRainfallMap[hourStr].total = record.total;
+    }
+  });
+
+  // 杞崲 map 鎴愭暟缁勫苟鎺掑簭
+  const hourlyRainfallList = Object.values(hourlyRainfallMap).sort((a, b) =>
+    a.time.localeCompare(b.time)
+  );
+
+  console.log("鉁� 鎸夊皬鏃惰仛鍚堝悗鐨勯檷闆ㄦ暟鎹細", hourlyRainfallList);
+
+  // 璁剧疆鍏ㄥ眬鍙橀噺
+  rainTotalInfo.value = hourlyRainfallList;
+
+  // 璁$畻鏃堕棿姝ラ暱
+  timeStepInfo = calculateTimeStep(rainTotalInfo.value);
+
+  // 鎻愬彇 intensity 鍊�
+  rainFallValues.value = hourlyRainfallList.map((r) => r.intensity);
+  // 璁$畻骞冲潎闆ㄥ己
+  if (rainFallValues.value.length > 0) {
+    const sumIntensity = rainFallValues.value.reduce(
+      (sum, val) => sum + val,
+      0
+    );
+    averageRainIntensity.value = sumIntensity / rainFallValues.value.length;
+  } else {
+    averageRainIntensity.value = 0; // 鎴栬�� null 琛ㄧず鏃犳暟鎹�
+  }
+
+  console.log("骞冲潎闆ㄥ己涓猴細", averageRainIntensity.value);
+
+  minRainValue.value = Math.min(...rainFallValues.value);
+  maxRainValue.value = Math.max(...rainFallValues.value);
+
+  console.log(
+    "褰撳墠鏂规涓嬫渶灏忛洦閲忓拰鏈�澶ч洦閲忥細",
+    minRainValue.value,
+    maxRainValue.value
+  );
+}
+// 瀹氫箟闄嶉洦绛夌骇鍙婂叾瀵瑰簲鐨勮瑙夊弬鏁�
+const rainLevels = [
+  {
+    name: "灏忛洦",
+    min: 0.1,
+    max: 9.9,
+    size: 0.5, // 闆ㄦ淮澶у皬锛氭洿灏�
+    speed: 20, // 涓嬭惤閫熷害锛氭洿鎱�
+    density: 15, // 闆ㄦ淮瀵嗗害锛氭洿绋�鐤�
+    color: "#ADD8E6", // 娴呰摑鑹诧紝璞″緛杞绘煍鐨勫皬闆�
+  },
+  {
+    name: "涓洦",
+    min: 10,
+    max: 24.9,
+    size: 0.6,
+    speed: 24,
+    density: 18,
+    color: "#ADD8E6",
+  },
+  {
+    name: "澶ч洦",
+    min: 25,
+    max: 49.9,
+    size: 0.7,
+    speed: 28,
+    density: 21,
+    color: "#ADD8E6",
+  },
+  {
+    name: "鏆撮洦",
+    min: 50,
+    max: 99.9,
+    size: 0.8,
+    speed: 32,
+    density: 24,
+    color: "#ADD8E6",
+  },
+  {
+    name: "澶ф毚闆�",
+    min: 100,
+    size: 0.9,
+    speed: 36,
+    density: 27,
+    color: "#ADD8E6",
+  },
+];
+// 鏍规嵁闄嶉洦閲忚繑鍥炲搴旂殑闆ㄥ舰閰嶇疆
+function getRainLevel(rainValue) {
+  for (let level of rainLevels) {
+    if (
+      level.min <= rainValue &&
+      (level.max === undefined || rainValue <= level.max)
+    ) {
+      return level;
+    }
+  }
+  // 榛樿鏃犻洦鐘舵��
+
+  return { name: "鏃犻洦", size: 0.3, speed: 10, density: 10, color: "#F0F8FF" };
+}
+// 鏍规嵁鎾斁杩涘害鏇存柊澶╂皵鏁堟灉锛堝凡浼樺寲锛�
+let lastUsedIndex = -1; // 缂撳瓨涓婁竴娆′娇鐢ㄧ殑绱㈠紩锛岄槻姝㈤噸澶嶆洿鏂�
+let lastRainValue = null;
+
+function calculateTimeStep(dataArray) {
+  if (!dataArray || dataArray.length < 2) {
+    console.warn("鏁版嵁涓嶈冻锛屾棤娉曡绠楁椂闂存闀�");
+    return null;
+  }
+
+  // 瑙f瀽鏃堕棿瀛楃涓蹭负 Date 瀵硅薄
+  function parseTime(timeStr) {
+    return new Date(timeStr.replace(" ", "T")); // 鍏煎 ISO 鏍煎紡
+  }
+
+  const firstTime = parseTime(dataArray[0].time);
+  const secondTime = parseTime(dataArray[1].time);
+
+  // 璁$畻鏃堕棿宸紙姣锛�
+  const diffMs = Math.abs(secondTime - firstTime);
+
+  // 杞崲涓哄皬鏃舵暟锛堜繚鐣欏皬鏁帮級
+  let timeStepHours = diffMs / (1000 * 60 * 60); // 姣 -> 灏忔椂
+
+  // 鍙�夛細閬嶅巻鎵�鏈夌浉閭婚」妫�鏌ユ槸鍚︿竴鑷�
+  for (let i = 1; i < dataArray.length - 1; i++) {
+    const current = parseTime(dataArray[i].time);
+    const next = parseTime(dataArray[i + 1].time);
+    const step = Math.abs(next - current) / (1000 * 60 * 60); // 姣 -> 灏忔椂
+    if (Math.abs(step - timeStepHours) > 0.01) {
+      console.warn(
+        `鍦ㄧ储寮� ${i} 澶勫彂鐜颁簡涓嶅悓鐨勬椂闂存闀�: ${step.toFixed(2)} 灏忔椂`
+      );
+    }
+  }
+
+  return timeStepHours;
+}
+// ============================================================================
+// 浼樺寲鏂瑰紡锛屽彲浠ユ眰鍑烘暣涓椂闂磋酱涓婏紝绗竴娆¢亣鍒拌繖鍏釜闃堝�煎緱鏃堕棿鐐癸紝鐒跺悗鍒嗘椂闂存鏄剧ず锛宎鏃堕棿鍐呮樉绀虹姸鎬�1锛岀劧鍚庣姸鎬佷氦鐣屽璁剧疆棰滆壊娓愬彉锛屽叾浣欏悓鐞嗭紝杩欐牱璺宠浆寰楁椂鍊欒兘澶熺洿鎺ヨ烦杞埌褰撳墠寰楅鑹蹭俊鎭樁娈碉紝鐩存帴搴旂敤锛屽嵆鍙�
+// ============================================================================
+// 鍏ㄥ眬鐘舵�佽褰�
+const colorState = {
+  currentColor: "#F5F0E6", // 褰撳墠棰滆壊
+  currentAlpha: -0.3, // 褰撳墠閫忔槑搴�
+  colorStages: null, // 棰勮绠楃殑棰滆壊闃舵鏃堕棿鐐�
+  maxColorTime: null, // 璁板綍杈惧埌鏈�娣遍鑹叉椂鐨勬椂闂寸偣
+};
+
+// 棰勮绠楅鑹查樁娈垫椂闂寸偣
+function precomputeColorStages() {
+  if (!rainTotalInfo.value || rainTotalInfo.value.length === 0) return;
+  // 棰滆壊閰嶇疆锛堜寒搴﹂�掑噺锛�
+  const COLOR_STOPS = [
+    { hex: "#F5F0E6", luminance: 240.4 }, // stage 0
+    { hex: "#E6D5B8", luminance: 214.8 }, // stage 1
+    { hex: "#D4B483", luminance: 184.0 }, // stage 2
+    { hex: "#B78B6A", luminance: 148.4 }, // stage 3
+    { hex: "#8B5A3A", luminance: 101.0 }, // stage 4
+    { hex: "#744C33", luminance: 84.5 }, // stage 5
+    { hex: "#5D3D2C", luminance: 68.1 }, // stage 6
+  ];
+  const alphaStops = [
+    -0.2, // stage 0
+    -0.3, // stage 1
+    -0.4, // stage 2
+    -0.5, // stage 3
+    -0.6, // stage 4
+    -0.7, // stage 5
+    -0.8, // stage 6
+  ];
+  // 绱闄嶉洦閲忛槇鍊硷紙mm锛�
+  const R_THRESHOLDS = [0, 200, 240, 280, 310, 350]; // 鍏�6涓樁娈靛搴�6涓槇鍊�
+  // 鏃堕棿鍜岄檷闆ㄩ噺淇℃伅
+  const timeTotals = [];
+  const initialTimestamp = new Date(rainTotalInfo.value[0].time).getTime();
+  for (let i = 0; i < rainTotalInfo.value.length; i++) {
+    const timestamp = new Date(rainTotalInfo.value[i].time).getTime();
+    const time = (timestamp - initialTimestamp) / 1000;
+    const total = rainTotalInfo.value[i].total; // 浣跨敤 total 鏇夸唬 intensity
+    timeTotals.push({
+      time,
+      total,
+    });
+  }
+
+  // 鎵惧嚭姣忎釜闃舵棣栨杈惧埌鐨勬椂闂寸偣
+  const stages = [];
+  for (let stage = 1; stage < R_THRESHOLDS.length + 1; stage++) {
+    const threshold = R_THRESHOLDS[stage - 1];
+    for (let i = 0; i < timeTotals.length; i++) {
+      const { time, total } = timeTotals[i];
+      if (total >= threshold) {
+        stages[stage] = {
+          startTime: time,
+          color: COLOR_STOPS[stage].hex,
+          alpha: alphaStops[stage],
+          threshold,
+        };
+        break;
+      }
+    }
+  }
+
+  // 濉厖闃舵0
+  stages[0] = {
+    startTime: 0,
+    color: COLOR_STOPS[0].hex,
+    alpha: alphaStops[0],
+    threshold: 0,
+  };
+
+  colorState.colorStages = stages;
+}
+
+function updateWaterColorByTime(isForceUpdate = false) {
+  if (!rainTotalInfo.value || rainTotalInfo.value.length === 0) return;
+
+  // 棣栨璋冪敤鏃堕璁$畻棰滆壊闃舵
+  if (colorState.colorStages === null) {
+    precomputeColorStages();
+  }
+
+  // 鏌ユ壘褰撳墠鏃堕棿鐐规墍灞炵殑闃舵
+  let currentStage = 0;
+  for (let i = colorState.colorStages.length - 1; i >= 0; i--) {
+    if (
+      colorState.colorStages[i] &&
+      currentTime.value >= colorState.colorStages[i].startTime
+    ) {
+      currentStage = i;
+      break;
+    }
+  }
+
+  // 璁板綍杈惧埌鏈�娣遍鑹茬殑鏃堕棿鐐�
+  if (currentStage >= colorState.colorStages.length - 1) {
+    if (
+      colorState.maxColorTime === null ||
+      currentTime.value > colorState.maxColorTime
+    ) {
+      colorState.maxColorTime = currentTime.value;
+    }
+  }
+
+  // 鍒ゆ柇鏄惁闇�瑕佸己鍒舵洿鏂伴鑹�
+  const isTimeGoingBackward = currentTime.value < colorState.lastTime;
+  const isBeforeMaxColorTime =
+    colorState.maxColorTime !== null &&
+    currentTime.value <= colorState.maxColorTime;
+  const shouldForceUpdate =
+    isForceUpdate && (isTimeGoingBackward || isBeforeMaxColorTime);
+
+  // 鏇存柊棰滆壊閫昏緫
+  if (shouldForceUpdate || isTimeGoingBackward) {
+    // 寮哄埗鏇存柊鎴栨椂闂村洖閫�鏃讹紝鐩存帴搴旂敤褰撳墠闃舵鐨勯鑹�
+    colorState.currentColor = colorState.colorStages[currentStage].color;
+    colorState.currentAlpha = colorState.colorStages[currentStage].alpha;
+  } else {
+    // 姝e父鏃堕棿鍓嶈繘鏃讹紝淇濇寔娓愯繘鍙樺寲
+    const newColor = colorState.colorStages[currentStage].color;
+    const newAlpha = colorState.colorStages[currentStage].alpha;
+
+    // 鍙簲鐢ㄦ洿鏆楃殑棰滆壊鍜屾洿浣庣殑閫忔槑搴�
+    if (
+      calculateLuminance(newColor) < calculateLuminance(colorState.currentColor)
+    ) {
+      colorState.currentColor = newColor;
+    }
+    if (newAlpha < colorState.currentAlpha) {
+      colorState.currentAlpha = newAlpha;
+    }
+  }
+
+  // 鏇存柊鏃堕棿璁板綍
+  colorState.lastTime = currentTime.value;
+  // ====== 鏂板锛氬湪 updateWaterColor 鍓嶆墦鍗板綋鍓嶄俊鎭� ======
+  // // 鑾峰彇褰撳墠绱闄嶉洦閲�
+  let currentTotal = null;
+  const baseTimestamp = new Date(rainTotalInfo.value[0].time).getTime();
+  const currentTimeMs = baseTimestamp + currentTime.value * 1000;
+
+  // 鎵惧埌鏈�鎺ヨ繎鐨勯檷闆ㄦ暟鎹偣
+  for (let i = rainTotalInfo.value.length - 1; i >= 0; i--) {
+    const dataTimeMs = new Date(rainTotalInfo.value[i].time).getTime();
+    if (dataTimeMs <= currentTimeMs) {
+      currentTotal = rainTotalInfo.value[i].total;
+      break;
+    }
+  }
+
+  // 鎵撳嵃淇℃伅
+  // console.log("========================================");
+  // console.log(`銆愭椂闂存埑銆�: ${new Date(currentTimeMs).toLocaleString()}`);
+  console.log(
+    `銆愮疮璁¢檷闆ㄩ噺 R銆�: ${
+      currentTotal !== null ? currentTotal.toFixed(2) : "鏈煡"
+    } mm`
+  );
+  // console.log(`銆愬綋鍓嶉樁娈点��: 绗� ${currentStage} 闃舵`);
+  console.log(
+    `銆愰鑹� HEX銆�: ${colorState.colorStages[currentStage]?.color || "鏈畾涔�"}`
+  );
+  // console.log(`銆愰�忔槑搴� Alpha銆�: ${colorState.colorStages[currentStage]?.alpha || '鏈畾涔�'}`);
+  // console.log("========================================");
+  // 搴旂敤棰滆壊
+  updateWaterColor(colorState.currentColor, colorState.currentAlpha);
+}
+
+// 杈呭姪鍑芥暟淇濇寔涓嶅彉
+function calculateLuminance(hex) {
+  const [r, g, b] = hexToRgb(hex);
+  return 0.299 * r + 0.587 * g + 0.114 * b;
+}
+
+function hexToRgb(hex) {
+  const bigint = parseInt(hex.slice(1), 16);
+  return [(bigint >> 16) & 255, (bigint >> 8) & 255, bigint & 255];
+}
+
+// 鏃堕棿杞磋烦杞嚱鏁�
+const seekToPosition = (event) => {
+  if (!isWaterPrimitiveCreated.value) {
+    ElMessage.warning("璇峰厛鍚姩姘翠綋妯℃嫙鍚庡啀杩涜鏃堕棿杞磋烦杞��");
+    return;
+  }
+
+  const rect = timelineTrack.value.getBoundingClientRect();
+  const percentage = (event.clientX - rect.left) / rect.width;
+  const targetTime = Math.round(percentage * duration.value);
+
+  const closestIndex = findClosestTimestampIndex(targetTime);
+  const baseTimestamp = waterTimestamps.value[0];
+  const newTime = (waterTimestamps.value[closestIndex] - baseTimestamp) / 1000;
+
+  // 鍒ゆ柇鏄惁闇�瑕佸己鍒舵洿鏂伴鑹�
+  const isGoingBackward = newTime < currentTime.value;
+  const isBeforeMaxColor =
+    colorState.maxColorTime !== null && newTime <= colorState.maxColorTime;
+  const shouldForceUpdate = isGoingBackward || isBeforeMaxColor;
+
+  currentTime.value = newTime;
+  setTimeForWaterSimulation(closestIndex);
+
+  // 鏍规嵁鏉′欢鏇存柊棰滆壊
+  updateWaterColorByTime(shouldForceUpdate);
+
+  if (!isPlaying.value) pauseWaterSimulation();
+};
+// ============================================================================
+
+function updateWeatherByProgress() {
+  if (rainFallValues.value.length === 0) return;
+  // console.log(`鏃堕棿杞存�绘椂闀�: ${duration.value}, 褰撳墠鏃堕棿: ${currentTime.value}`); // 鎵撳嵃鏃堕棿杞翠俊鎭�
+  const progress = currentTime.value / duration.value;
+  const floatIndex = progress * (rainFallValues.value.length - 1);
+  const index = Math.floor(floatIndex); // 褰撳墠绱㈠紩
+  const nextIndex = Math.min(index + 1, rainFallValues.value.length - 1); // 涓嬩竴绱㈠紩
+  const currentRain = rainFallValues.value[index];
+  const nextRain = rainFallValues.value[nextIndex];
+  // 鍚敤鎻掑�硷紙alpha 骞虫粦杩囨浮锛�
+  const alpha = floatIndex - index;
+  // const rainValue = currentRain + (nextRain - currentRain) * alpha;
+  const rainValue = currentRain + (nextRain - currentRain);
+  // 鎵撳嵃褰撳墠澶勭悊鐨勯洦閲忔暟鎹�
+  // console.log(
+  //   `姝e湪澶勭悊鐨勯洦閲忔暟鎹偣: 褰撳墠=${currentRain}, 涓嬩竴涓�=${nextRain}, 鎻掑�煎悗=${rainValue.toFixed(
+  //     2
+  //   )}, 绱㈠紩=${index}`
+  // );
+  // 濡傛灉褰撳墠绱㈠紩鏈彉鍖栦笖鎻掑�煎樊寮備笉澶э紝璺宠繃閲嶅鏇存柊
+  if (index === lastUsedIndex && Math.abs(rainValue - lastRainValue) < 0.1) {
+    // console.log('鐢变簬鏁版嵁鏃犳樉钁楀彉鍖栵紝璺宠繃鏈鏇存柊');
+    return;
+  }
+
+  lastUsedIndex = index;
+  lastRainValue = rainValue;
+
+  // 鑾峰彇瀵瑰簲鐨勯洦褰㈤厤缃�
+  const rainLevel = getRainLevel(rainValue);
+
+  // if (rainLevel.name === '鏃犻洦') {
+  //   // 鏃犻洦鐘舵�侊細娓呴櫎闆ㄦ晥
+  //   mapUtils.delRain();
+  //   console.log('鎵ц浜嗘棤闆ㄧ姸鎬侊紝娓呴櫎浜嗛洦鏁�');
+  //   return;
+  // }
+
+  // 闈炴棤闆ㄧ姸鎬侊細鏋勫缓闆ㄦ淮鍙傛暟骞舵洿鏂伴洦鏁�
+  const rainParams = {
+    rainSize: rainLevel.size,
+    rainSpeed: rainLevel.speed,
+    rainDensity: rainLevel.density,
+    rainColor: rainLevel.color,
+  };
+  console.log("褰撳墠闆ㄩ噺鏁版嵁锛�", rainValue, "褰撳墠闆ㄥ舰锛�", rainLevel);
+  // 璋冪敤宸ュ叿鏂规硶鏇存柊闆ㄦ晥
+  mapUtils.toggleRain(rainParams, true);
+}
+const stopPlayback = () => {
+  clearInterval(playInterval);
+};
+
+const skipForward = () => {
+  if (waterTimestamps.value.length === 0) return;
+  const currentIndex = findClosestTimestampIndex(currentTime.value);
+  const nextIndex = currentIndex + 1;
+  if (nextIndex >= waterTimestamps.value.length) {
+    return;
+  }
+  const baseTimestamp = waterTimestamps.value[0];
+  currentTime.value = (waterTimestamps.value[nextIndex] - baseTimestamp) / 1000;
+  setTimeForWaterSimulation(nextIndex);
+  if (!isPlaying.value) pauseWaterSimulation();
+};
+
+const skipBackward = () => {
+  if (waterTimestamps.value.length === 0) return;
+  const currentIndex = findClosestTimestampIndex(currentTime.value);
+  const prevIndex = currentIndex - 1;
+  if (prevIndex < 0) {
+    return;
+  }
+  const baseTimestamp = waterTimestamps.value[0];
+  currentTime.value = (waterTimestamps.value[prevIndex] - baseTimestamp) / 1000;
+  setTimeForWaterSimulation(prevIndex);
+  if (!isPlaying.value) pauseWaterSimulation();
+};
+const toggleSpeedMenu = () => (showSpeedMenu.value = !showSpeedMenu.value);
+
+// 璁剧疆鎾斁閫熺巼
+const setPlaybackRate = (rate) => {
+  isColorRenderEnabled.value = false;
+  playbackRate.value = rate;
+  showSpeedMenu.value = false;
+  // 鍋滄褰撳墠鎾斁
+  stopPlayback();
+  setTimeout(() => {
+    mapUtils.delRain();
+  }, 3000);
+  // 閲嶇疆鏃堕棿杞村埌鍒濆鐘舵��
+  currentTime.value = 0; // 鏃堕棿褰掗浂
+  emit("timeUpdate", progressPercentage.value);
+  isPlaying.value = false;
+  emit("isPlaying", false);
+
+  // 閿�姣佺幇鏈夌殑姘翠綋妯℃嫙灞�
+  if (isWaterPrimitiveCreated.value) {
+    destoryWaterPrimitive();
+    isWaterPrimitiveCreated.value = false; // 閲嶇疆鏍囧織鍙橀噺
+  }
+  isPlaying.value = false;
+  emit("isPlaying", false);
+
+  pauseWaterSimulation(); // 璋冪敤鏆傚仠鎺ュ彛
+  EventBus.emit("clear-echart");
+  EventBus.emit("reset-table");
+};
+
+// 杈呭姪鍑芥暟锛氭壘鍒版渶鎺ヨ繎鐨勬椂闂存埑绱㈠紩
+function findClosestTimestampIndex(currentTimeValue) {
+  if (waterTimestamps.value.length === 0) return 0;
+
+  // 璁$畻褰撳墠鏃堕棿瀵瑰簲鐨勬绉掓椂闂存埑
+  const baseTime = waterTimestamps.value[0];
+  const currentTimestamp = baseTime + currentTimeValue * 1000;
+
+  // 鎵惧埌鏈�鎺ヨ繎鐨� timestamp 绱㈠紩
+  let closestIndex = 0;
+  let minDiff = Infinity;
+
+  waterTimestamps.value.forEach((timestamp, index) => {
+    const diff = Math.abs(timestamp - currentTimestamp);
+    if (diff < minDiff) {
+      minDiff = diff;
+      closestIndex = index;
+    }
+  });
+
+  return closestIndex;
+}
+watch(
+  () => selectedScheme.value,
+  (newVal) => {
+    if (newVal) {
+      console.log("閫変腑鏂规宸叉敼鍙�:", newVal);
+    }
+  }
+);
+watch(
+  () => currentTime.value,
+  () => {
+    if (waterTimestamps.value.length > 0) {
+      sendCurrentPlayingTime.value = dayjs(waterTimestamps.value[0])
+        .add(currentTime.value, "second")
+        .valueOf(); // 浣跨敤 valueOf() 鑾峰彇鍘熷鏃堕棿鎴�
+
+      // 鏇存柊 currentPlayingTime 鏍煎紡鍖栧悗鐨勬椂闂村瓧绗︿覆
+      currentPlayingTime.value = dayjs(sendCurrentPlayingTime.value).format(
+        "YYYY-MM-DD HH:mm:ss"
+      );
+      EventBus.emit("time-update", currentPlayingTime.value);
+    }
+  }
+);
+
+// 鏃堕棿鏍囪鐢熸垚
+function generateTimeMarkers(timestamps) {
+  if (!timestamps || timestamps.length === 0) return [];
+  const sorted = [...timestamps].sort((a, b) => dayjs(a).diff(dayjs(b)));
+  const interval = Math.floor(
+    dayjs(sorted.at(-1)).diff(dayjs(sorted[0]), "second") / 4
+  );
+  return Array.from({ length: 5 }, (_, i) =>
+    dayjs(sorted[0])
+      .add(i * interval, "second")
+      .format("MM-DD HH:mm:ss")
+  );
+}
+
+watch(
+  () => waterTimestamps.value,
+  (newTimestamps) => {
+    if (newTimestamps.length > 0)
+      timeMarkers.value = generateTimeMarkers(newTimestamps);
+  },
+  { immediate: true }
+);
+
+const jsonFetch = ref(null);
+const currentReplayIndex = ref(0); // 褰撳墠鎾斁鐨剅ePlayList绱㈠紩
+// 鎻愬彇涓虹嫭绔嬪嚱鏁�
+async function initializeSimulationData(replayItem = null) {
+  try {
+    const schemeInfo = selectedScheme.value;
+    serviceInfo = schemeInfo.serviceName;
+
+    if (schemeInfo.type == 2) {
+      if (
+        replayItem ||
+        (simStore.rePlayList && simStore.rePlayList.length != 0)
+      ) {
+        jsonFetch.value =
+          replayItem || simStore.rePlayList[currentReplayIndex.value];
+        speedShow.value = true;
+      } else {
+        jsonFetch.value = layerDate.value;
+        speedShow.value = false;
+      }
+    } else {
+      getRainfallData();
+      speedShow.value = true;
+      jsonFetch.value = null;
+    }
+
+    // console.log('鑾峰彇鍒扮殑 serviceName:', serviceInfo);
+    // 鏍规嵁 layer.json 鑾峰彇鏃堕棿杞翠俊鎭�
+    const {
+      waterTimestamps: timestamps,
+      watersMaxHeight,
+      watersMinHeight,
+    } = await fetchWaterSimulationData(serviceInfo, jsonFetch.value);
+
+    console.log(
+      "褰撳墠鏂规涓嬬殑鏈�澶ф按浣嶆繁搴﹀拰鏈�灏忔按浣嶆繁搴�",
+      watersMaxHeight,
+      watersMinHeight
+    );
+    const waterInfoArr = [
+      watersMaxHeight,
+      maxRainValue.value,
+      averageRainIntensity.value,
+    ];
+    schemWaterInfo.value = waterInfoArr;
+    // 鏇存柊鏃堕棿杞寸浉鍏虫暟鎹�
+    if (timestamps) {
+      frameNum.value = timestamps.length;
+      waterTimestamps.value = timestamps;
+      updateTimelineRange();
+      timeMarkers.value = generateTimeMarkers(timestamps);
+      sendCurrentPlayingTime.value = timestamps[0];
+      currentPlayingTime.value = dayjs(timestamps[0]).format(
+        "YYYY-MM-DD HH:mm:ss"
+      );
+    }
+    minFlowRate = watersMinHeight;
+    maxFlowRate = watersMaxHeight;
+  } catch (error) {
+    console.error("Error loading water simulation data:", error);
+    ElMessage({
+      message: "闄嶉洦鏁版嵁鍑洪敊锛岃閲嶆柊鏂板缓妯℃嫙鏂规锛�",
+      type: "warning",
+    });
+  }
+}
+// 鎾斁瀹屾垚鍚庣殑鍥炶皟
+function handlePlayFinished() {
+  if (selectedScheme.value.type !== 2) return;
+  finishPlay.value = false;
+  currentReplayIndex.value++;
+
+  if (currentReplayIndex.value < simStore.rePlayList.length) {
+    // 鑷姩鎾斁涓嬩竴涓�
+    initializeSimulationData(simStore.rePlayList[currentReplayIndex.value]);
+    togglePlay();
+    shouldAutoPlay.value = false;
+  } else {
+    // 鎵�鏈夐」鐩挱鏀惧畬鎴�
+    currentReplayIndex.value = 0; // 閲嶇疆绱㈠紩
+    isPlaying.value = false; // 鍋滄鎾斁
+    emit("timeUpdate", 100); // 鍦ㄦ墍鏈夐」鐩挱鏀惧畬姣曞悗瑙﹀彂
+  }
+}
+
+// 鐩戝惉鎾斁瀹屾垚浜嬩欢
+watch(
+  () => finishPlay.value,
+  (newVal) => {
+    if (
+      newVal &&
+      selectedScheme.value.type === 2 &&
+      simStore.rePlayList.length > 0
+    ) {
+      handlePlayFinished();
+    }
+  }
+);
+
+const shouldAutoPlay = ref(false);
+// 鐩戝惉 layerDate 鍙樺寲鍚庢爣璁板噯澶囨挱鏀�
+watch(
+  () => layerDate.value,
+  async (newVal) => {
+    if (selectedScheme.value.type === 2 && newVal) {
+      shouldAutoPlay.value = true;
+    }
+  },
+  { deep: true }
+);
+// 绛夊緟 finishPlay 鎴愬姛鍚庡啀鎾斁
+watchEffect(() => {
+  if (shouldAutoPlay.value && finishPlay.value && !isPlaying.value) {
+    initializeSimulationData();
+    togglePlay();
+    shouldAutoPlay.value = false;
+  }
+});
+
+// 鎸傝浇鏃惰皟鐢�
+onMounted(async () => {
+  // 鍥犱负杩欎釜鍑芥暟瀹炴椂妯℃嫙鐩戝惉涔熼渶瑕佷娇鐢紝鎵�浠ュ皝瑁呬簡涓�涓嚱鏁�
+  await initializeSimulationData();
+});
+
+// 鏍规嵁杩斿洖鏁版嵁鐨勪釜鏁板幓娓叉煋鏃堕棿杞�
+function updateTimelineRange() {
+  if (waterTimestamps.value.length > 0) {
+    const [first, last] = [
+      waterTimestamps.value[0],
+      waterTimestamps.value.at(-1),
+    ];
+    duration.value = (last - first) / 1000; // 姣杞
+  }
+}
+onBeforeUnmount(() => {
+  stopPlayback();
+  destoryWaterPrimitive();
+});
+
+const { endSimulate } = inject("simulateActions");
+
+async function handleBack() {
+  // 瀹炴椂妯℃嫙寮圭獥纭鏄繑鍥炴柟妗堝垪琛ㄨ繕鏄仠姝㈡ā鎷�
+  if (selectedScheme.value.type === 2) {
+    try {
+      await ElMessageBox.confirm("鏂规鏈仠姝㈡椂缁撴潫妯℃嫙鍚庯紝鍚庡彴灏嗗仠姝㈣绠�", {
+        confirmButtonText: "缁撴潫妯℃嫙",
+        cancelButtonText: "杩斿洖鍒楄〃",
+        type: "warning",
+      });
+
+      const res = await stopSim(selectedScheme.value.id);
+      if (res.code == 404) {
+        ElMessage.warning("璇ユ湇鍔″凡鍋滄");
+      } else {
+        ElMessage.success("鏈嶅姟姝e湪鍋滄涓�");
+      }
+    } catch (error) {
+      // 鐢ㄦ埛鐐瑰嚮浜嗐�愯繑鍥炲垪琛ㄣ�戞垨鑰呭嚭鐜伴敊璇�
+      return;
+    }
+  }
+
+  // 涓嶇 type 鏄笉鏄� 2锛屾渶缁堥兘鎵ц缁撴潫妯℃嫙鎿嶄綔
+  endSimulation();
+}
+
+async function endSimulation() {
+  clearAllPoints();
+  simStore.openDia = true;
+  simStore.crossSection = [];
+  // 缁撴潫妯℃嫙涔嬪悗娓呴櫎layer鍒楄〃
+  simStore.rePlayList = [];
+  EventBus.emit("close-time");
+  endSimulate();
+  isWaterPrimitiveCreated.value = false;
+
+  // 缁撴潫璁$畻鍜屽仠姝㈡嬀鍙�
+  if (ratelevelRef.value) {
+    ratelevelRef.value.endCalculation();
+    ratelevelRef.value.stopPicking();
+  }
+
+  // 娓呴櫎鐐�
+  if (crossRef.value) {
+    crossRef.value.clearPoints();
+    console.log("鎵ц鍒犻櫎鐐瑰姛鑳�");
+  }
+
+  emit("isColorRender", false);
+
+  // 寤惰繜鍒犻櫎闆ㄩ噺鍥惧眰
+  setTimeout(() => {
+    mapUtils.delRain();
+  }, 3000);
+
+  destoryWaterPrimitive();
+
+  // 鍙戦�佷簨浠堕殣钘忕浉鍏充俊鎭�
+  EventBus.emit("hide-schemeInfo");
+  EventBus.emit("clear-water-depth");
+  EventBus.emit("clear-water-velocity");
+
+  ElMessage({ message: "妯℃嫙杩涚▼姝e湪鍏抽棴涓�...", type: "success" });
+}
+</script>
+<style scoped>
+.timeline-container {
+  display: flex;
+  /* align-items: center; */
+  justify-content: space-between;
+  position: absolute;
+  bottom: 10%;
+  left: 50%;
+  transform: translateX(-50%);
+  z-index: 99;
+  width: 44%;
+  height: 10%;
+  /* background-color: #1a2634; */
+  background: url("@/assets/img/menubar/bar.png");
+  background-size: 100% 100%;
+  color: white;
+  /* border-radius: 8px; */
+  font-family: Arial, sans-serif;
+  padding: 0 25px;
+}
+
+.controls {
+  display: flex;
+  /* align-items: center; */
+  margin: 25px 25px 10px 0;
+}
+
+.control-btn {
+  background: none;
+  border: none;
+  color: white;
+  font-size: 16px;
+  cursor: pointer;
+  margin-right: 10px;
+  width: 30px;
+  height: 30px;
+  display: flex;
+  align-items: center;
+  justify-content: center;
+  border-radius: 50%;
+
+  img {
+    width: 100%;
+    height: 100%;
+  }
+}
+
+.control-btn:hover {
+  background-color: rgba(255, 255, 255, 0.1);
+}
+
+.play-btn {
+  background-color: #4a90e2;
+  width: 36px;
+  height: 36px;
+}
+
+.speed-control {
+  position: relative;
+  cursor: pointer;
+  /* padding: 5px 10px; */
+  border-radius: 4px;
+  background-color: rgba(255, 255, 255, 0.1);
+  width: 36px;
+  height: 36px;
+  line-height: 36px;
+  text-align: center;
+}
+
+.speed-menu {
+  position: absolute;
+  top: 100%;
+  left: 0;
+  background-color: #2a3a4a;
+  border-radius: 4px;
+  z-index: 10;
+  width: 60px;
+}
+
+.speed-menu div {
+  /* padding: 5px 5px; */
+  text-align: center;
+}
+
+.speed-menu div:hover,
+.speed-menu div.active {
+  /* background-color: #4a90e2; */
+  background-color: rgba(127, 255, 212, 0.5);
+}
+
+.timeline {
+  margin-top: 10px;
+  position: relative;
+  flex: 0.9;
+}
+
+.dates {
+  display: flex;
+  justify-content: space-between;
+  margin-bottom: 5px;
+}
+
+.date-label {
+  font-size: 14px;
+  color: #fff;
+}
+
+.timeline-track {
+  height: 8px;
+  background-color: rgba(255, 255, 255, 0.1);
+  border-radius: 4px;
+  position: relative;
+  cursor: pointer;
+}
+
+.timeline-progress {
+  height: 100%;
+  background-color: #4a90e2;
+  border-radius: 4px;
+  position: absolute;
+  top: 0;
+  left: 0;
+}
+
+.timeline-cursor {
+  width: 12px;
+  height: 12px;
+  background-color: white;
+  border-radius: 50%;
+  position: absolute;
+  top: 50%;
+  transform: translate(-50%, -50%);
+  z-index: 2;
+}
+
+.time-markers {
+  position: absolute;
+  width: 100%;
+  top: 20px;
+  display: flex;
+  justify-content: space-between;
+}
+
+.time-marker {
+  position: absolute;
+  font-size: 12px;
+  color: #fff;
+  text-align: center;
+  width: 25%;
+  white-space: nowrap;
+  display: flex;
+  flex-direction: column;
+  align-items: center;
+}
+
+/* .date-part {
+  margin-bottom: 2px;
+} */
+.time-part {
+  font-size: 11px;
+  opacity: 0.8;
+}
+
+.current-date {
+  margin-bottom: 5px;
+  font-size: 15px;
+  color: #fff;
+  transform: translateX(-8%);
+}
+
+.scale-markers {
+  position: absolute;
+  width: 100%;
+  height: 10px;
+  top: 10px;
+}
+
+.scale-marker {
+  position: absolute;
+  width: 3px;
+  height: 6px;
+  background-color: rgba(255, 255, 255, 1);
+  transform: translateX(-50%);
+}
+</style>

--
Gitblit v1.9.3