timeLine.vue 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246
  1. <template>
  2. <div class="time-line" v-show="isShow">
  3. <div class="play" @click="handleChange">
  4. <img
  5. class="icon"
  6. :src="require(`@/assets/images/warningHome/${isCounting ? 'transparent-pause-icon.png' : 'transparent-play-icon.png'}`)"
  7. alt=""
  8. />
  9. </div>
  10. <div class="line">
  11. <div class="active-line" :style="{ width: numList[active] + '%' }"></div>
  12. <div :class="['dot-item', { mr: index === 6 }]" v-for="(item, index) in list" :key="index">
  13. <div :class="['dot', { active: active === index }]"></div>
  14. <span :class="{ text: active === index }">{{ item }}</span>
  15. </div>
  16. </div>
  17. </div>
  18. </template>
  19. <script setup>
  20. import { ref, onDeactivated,computed ,watch, onActivated} from "vue";
  21. import { useStore } from "vuex";
  22. import eventBus from "@/api/eventBus";
  23. const store = useStore();
  24. const isShow = ref(true)
  25. // const explainParams = computed(() => store.state.home.explainData);
  26. // // 监听 periodId 的变化
  27. // watch(() => explainParams.value, (newValue, oldValue) => {
  28. // isShow.value = false
  29. // if (newValue.id) {
  30. // isShow.value = true
  31. // }
  32. // });
  33. onActivated(()=>{
  34. restActive()
  35. })
  36. // 果园切换监听事件
  37. eventBus.on("select:changeGarden", (e) => {
  38. restActive()
  39. });
  40. //监听测报面板高度变化
  41. eventBus.on('report:shrink',(e)=>{
  42. if(e>=Math.round(0.7 * window.innerHeight)){
  43. clearTime();
  44. active.value = 0;
  45. isCounting.value = false;
  46. eventBus.emit("weatherTime:resetTime")
  47. }else{
  48. restActive()
  49. }
  50. })
  51. const active = ref(0);
  52. const isCounting = ref(false);
  53. const numList = [7, 21, 37, 52, 68, 84, 99];
  54. function formatDateToMMDD(date) {
  55. const month = String(date.getMonth() + 1).padStart(2, "0"); // 月份从0开始,需要加1,并确保是两位数
  56. const day = String(date.getDate()).padStart(2, "0"); // 确保日期是两位数
  57. return `${month}/${day}`;
  58. }
  59. function getCurrentDateAndNextSixDaysMMDD() {
  60. const currentDate = new Date();
  61. const dates = [];
  62. for (let i = 0; i <= 6; i++) {
  63. const newDate = new Date(currentDate);
  64. newDate.setDate(currentDate.getDate() + i);
  65. const formattedDate = formatDateToMMDD(newDate);
  66. dates.push(formattedDate);
  67. }
  68. return dates;
  69. }
  70. const list = getCurrentDateAndNextSixDaysMMDD();
  71. const incrementInterval = 3000; // 默认间隔3秒
  72. const specialIncrementInterval = 3000; // 特殊间隔5秒
  73. const timer = ref(null);
  74. const incrementCount = (type) => {
  75. if (active.value === 6) {
  76. active.value = 0;
  77. eventBus.emit("weatherTime:changeTime",active.value)
  78. } else {
  79. if(type!=='original'){
  80. active.value += 1;
  81. eventBus.emit("weatherTime:changeTime",active.value)
  82. }
  83. }
  84. let time = incrementInterval;
  85. if (active.value === 0 || active.value === 6) {
  86. time = specialIncrementInterval;
  87. }
  88. timer.value = setTimeout(() => {
  89. if (isCounting.value) incrementCount();
  90. }, time);
  91. };
  92. const timerId = ref(null);
  93. const handleChange = () => {
  94. clearTime()
  95. if (active.value === 0 || active.value === 6) {
  96. timerId.value = setTimeout(() => {
  97. if (isCounting.value) incrementCount();
  98. }, specialIncrementInterval);
  99. } else {
  100. incrementCount('original');
  101. }
  102. isCounting.value = !isCounting.value;
  103. if (isCounting.value) {
  104. eventBus.emit("weatherTime:changeTime", 0)
  105. }
  106. };
  107. onDeactivated(() => {
  108. clearTime();
  109. });
  110. //清除定时器
  111. const clearTime = () => {
  112. if (timerId.value) {
  113. clearTimeout(timerId.value);
  114. }
  115. if (timer.value) {
  116. clearTimeout(timer.value);
  117. }
  118. };
  119. const restActive = () => {
  120. clearTime();
  121. active.value = 0;
  122. isCounting.value = false;
  123. handleChange();
  124. };
  125. eventBus.on("map_click_alarm", ({warningTypeId,factorId})=>{
  126. restActive()
  127. })
  128. defineExpose({ restActive });
  129. </script>
  130. <style lang="scss" scoped>
  131. .time-line {
  132. width: 100%;
  133. height: 100%;
  134. border-radius: 10px 10px 20px 20px;
  135. background: rgba(35, 35, 35, 0.6);
  136. border: 1px solid rgba(255,255,255,0.4);
  137. display: flex;
  138. align-items: center;
  139. box-sizing: border-box;
  140. padding: 0 20px;
  141. .play {
  142. display: flex;
  143. align-items: center;
  144. margin-right: 17px;
  145. .icon {
  146. width: 46px;
  147. }
  148. }
  149. .line {
  150. background: linear-gradient(30deg, #fff 0% , rgba(44, 44, 44,0.6) 100%);
  151. border-radius: 2px;
  152. width: 80%;
  153. height: 2px;
  154. display: flex;
  155. justify-content: space-between;
  156. position: relative;
  157. z-index: 2;
  158. position: relative;
  159. top: -4px;
  160. .active-line {
  161. background: linear-gradient(30deg, #eccd9b 0% ,#F3C11D 100%,);
  162. position: absolute;
  163. top: 0;
  164. left: 0;
  165. height: 2px;
  166. z-index: 1;
  167. }
  168. .dot-item {
  169. color: #fff;
  170. font-size: 16px;
  171. position: relative;
  172. z-index: 2;
  173. .dot {
  174. width: 8px;
  175. height: 8px;
  176. background: #E6E6E6;
  177. border-radius: 50%;
  178. margin: -2.2px 0 6px 14px;
  179. position: relative;
  180. &::after{
  181. content: '';
  182. position: absolute;
  183. top: -2px;
  184. left: -2px;
  185. width: 8px;
  186. height: 8px;
  187. border-radius: 50%;
  188. border: 2px solid rgba(255,255,255,0.2);
  189. }
  190. }
  191. span {
  192. margin-left: 3px;
  193. letter-spacing: 0.5px;
  194. }
  195. .active {
  196. width: 6px;
  197. height: 6px;
  198. margin: -2px 0 5px 14px;
  199. position: relative;
  200. background: #F3C11D;
  201. &::after{
  202. content: '';
  203. position: absolute;
  204. top: -2px;
  205. left: -2px;
  206. width: 6px;
  207. height: 6px;
  208. border-radius: 50%;
  209. border: 2px solid rgba(243,193,29,0.2);
  210. }
  211. }
  212. .text {
  213. // color: #F3C11D;
  214. background: #ED9E1E;
  215. padding: 1px 10px;
  216. border-radius: 36px;
  217. }
  218. }
  219. .mr {
  220. margin-right: -12px;
  221. }
  222. }
  223. }
  224. </style>