zheng 1 天之前
父节点
当前提交
2891c2abed

+ 18 - 8
vue3/project4/src/router/index.ts

@@ -1,17 +1,21 @@
 import { createRouter, createWebHistory, createWebHashHistory } from 'vue-router';
-import type {RouteRecordRaw} from 'vue-router'
+import type { RouteRecordRaw } from 'vue-router'
 // 1.引入页面路径
 import Home from '@/views/Home.vue';
 import List from '@/views/List.vue';
 import My from '@/views/My.vue';
+import { KeepAlive } from 'vue';
 // import Demo1 from '@/views/Demo1.vue';
 // import Demo2 from '@/views/Demo2.vue';
-
+interface routeMeta {
+    KeepAlive: boolean
+}
 type RouteType = RouteRecordRaw & {
-    children?: RouteType[]
+    children?: RouteType[],
+    meta?: routeMeta
 }
 
-const routes:RouteType[] = [
+const routes: RouteType[] = [
     // 当前项目匹配的路由
     // 2.匹配路由
     {
@@ -31,14 +35,20 @@ const routes:RouteType[] = [
             {
                 path: 'demo1',
                 // path: 'demo1/:id/:user/:sex',
-                component: ()=>import("../views/Demo1.vue"),
-                
+                component: () => import("../views/Demo1.vue"),
+                meta: {
+                    KeepAlive: true
+                }
+
             },
             {
                 // path: 'demo2/:id/:user/:sex',
                 path: 'demo2',
-                name:'dier',
-                component: ()=>import("../views/Demo2.vue"),
+                name: 'dier',
+                component: () => import("../views/Demo2.vue"),
+                meta: {
+                    KeepAlive: false
+                }
             }
         ]
     },

+ 1 - 0
vue3/project4/src/views/Demo1.vue

@@ -2,6 +2,7 @@
   <div>
     demo1
     <p>我叫{{ user }},是个{{ sex }}孩,我的编号是{{ id }}</p>
+    <input type="text" name="" id="">
   </div>
 </template>
 

+ 1 - 0
vue3/project4/src/views/Demo2.vue

@@ -2,6 +2,7 @@
   <div>
     demo2
     <p>我叫{{ user }},是个{{ sex }}孩,我的编号是{{ id }}</p>
+    <input type="text" name="" id="">
   </div>
 </template>
 

+ 15 - 4
vue3/project4/src/views/List.vue

@@ -9,7 +9,7 @@
       <!-- <li v-for="(item) in list" :key="item.id">
         <RouterLink :to="`/list/demo1?id=${item.id}&user=${item.name}&sex=${item.sex}`">{{ item.name }}</RouterLink>
       </li> -->
-      <li v-for="(item) in list" :key="item.id">
+      <!-- <li v-for="(item) in list" :key="item.id">
         <RouterLink :to="{
           name:'dier',
           query:{
@@ -18,18 +18,29 @@
             sex:item.sex
           }
         }">{{ item.name }}</RouterLink>
-      </li>
+      </li> -->
     </ul>
+    <RouterLink to="/list/demo1">第一个</RouterLink>
+    <RouterLink to="/list/demo2">第二个</RouterLink>
     <div id="main">
-      <RouterView></RouterView>
+      <!-- <RouterView></RouterView> -->
+      <RouterView v-slot="{Component}">
+        <KeepAlive>
+          <component :is="Component" v-if="route.meta.KeepAlive"></component>
+        </KeepAlive>
+          <component :is="Component" v-if="!route.meta.KeepAlive"></component>
+      </RouterView>
+
     </div>
   </div>
 </template>
 
 <script lang="ts" setup>
 import {ref,reactive} from "vue" 
-import { RouterView,RouterLink } from "vue-router";
+import { RouterView,RouterLink,useRoute } from "vue-router";
 import { nanoid } from "nanoid";
+const route = useRoute();
+console.log(route,'哈哈哈')
 // interface
 let list = reactive([
   {

+ 376 - 1
vue3/project4/src/views/My.vue

@@ -1,4 +1,4 @@
-<template>
+<!-- <template>
   <div>我的</div>
 </template>
 
@@ -8,4 +8,379 @@ import {ref,reactive} from "vue"
 </script>
 
 <style lang="scss" scoped>
+</style> -->
+<template>
+  <div class="weather-app">
+    
+    <!-- 搜索区域 -->
+    <div class="search-container">
+      <input
+        type="text"
+        v-model="searchQuery"
+        placeholder="输入城市名称..."
+        @keyup.enter="searchWeather"
+      >
+      <button @click="searchWeather">查询</button>
+    </div>
+    
+    <!-- 加载状态 -->
+    <div class="loading" v-if="isLoading">
+      正在查询天气数据...
+    </div>
+    
+    <!-- 错误提示 -->
+    <div class="error" v-if="errorMessage">
+      ⚠️ {{ errorMessage }}
+    </div>
+    
+    <!-- 天气信息展示 -->
+    <div class="weather-card" v-if="currentWeather && !isLoading && !errorMessage">
+      <div class="weather-header">
+        <h2>{{ currentWeather.city }}</h2>
+        <div class="date">{{ formattedDate }}</div>
+      </div>
+      
+      <div class="weather-main">
+        <div class="temperature">
+          <span class="value">{{ currentTemp }}</span>
+          <span class="unit" @click="toggleUnit">
+            {{ isCelsius ? '°C' : '°F' }}
+          </span>
+        </div>
+        
+        <div class="condition">
+          <div class="icon">{{ weatherIcon }}</div>
+          <div class="text">{{ currentWeather.condition }}</div>
+        </div>
+      </div>
+      
+      <div class="weather-details">
+        <div class="detail">
+          <span class="label">湿度:</span>
+          <span class="value">{{ currentWeather.humidity }}%</span>
+        </div>
+        <div class="detail">
+          <span class="label">风速:</span>
+          <span class="value">{{ currentWeather.windSpeed }} km/h</span>
+        </div>
+        <div class="detail">
+          <span class="label">气压:</span>
+          <span class="value">{{ currentWeather.pressure }} hPa</span>
+        </div>
+      </div>
+    </div>
+    
+    <!-- 历史记录 -->
+    <div class="history-section" v-if="searchHistory.length > 0">
+      <h3>查询历史</h3>
+      <div class="history-tags">
+        <span 
+          class="history-tag" 
+          v-for="(item, index) in searchHistory" 
+          :key="index"
+          @click="searchWeather(item)"
+        >
+          {{ item }}
+          <button @click.stop="removeHistory(index)" class="remove-btn">×</button>
+        </span>
+      </div>
+    </div>
+  </div>
+</template>
+
+<script setup lang="ts">
+import { ref, computed, watch } from 'vue';
+
+// 1. 定义天气数据类型接口
+interface WeatherData {
+  city: string;
+  temperature: number; // 摄氏度
+  condition: string; // 天气状况
+  humidity: number; // 湿度百分比
+  windSpeed: number; // 风速 km/h
+  pressure: number; // 气压 hPa
+  date: Date; // 日期
+}
+
+// 2. 响应式状态
+const searchQuery = ref<string>('');
+const currentWeather = ref<WeatherData | null>(null);
+const isLoading = ref<boolean>(false);
+const errorMessage = ref<string>('');
+const isCelsius = ref<boolean>(true);
+const searchHistory = ref<string[]>([]);
+
+// 3. 温度单位转换计算属性
+const currentTemp = computed<number>(() => {
+  if (!currentWeather.value) return 0;
+  
+  // 如果是华氏度,进行转换 (°F = °C × 1.8 + 32)
+  return isCelsius.value 
+    ? currentWeather.value.temperature 
+    : Math.round((currentWeather.value.temperature * 1.8 + 32) * 10) / 10;
+});
+
+// 4. 格式化日期
+const formattedDate = computed<string>(() => {
+  if (!currentWeather.value) return '';
+  
+  return currentWeather.value.date.toLocaleDateString('zh-CN', {
+    year: 'numeric',
+    month: 'long',
+    day: 'numeric',
+    weekday: 'long'
+  });
+});
+
+// 5. 根据天气状况显示对应图标
+const weatherIcon = computed<string>(() => {
+  if (!currentWeather.value) return '';
+  
+  const condition = currentWeather.value.condition.toLowerCase();
+  
+  if (condition.includes('晴')) return '☀️';
+  if (condition.includes('雨')) return '🌧️';
+  if (condition.includes('云')) return '☁️';
+  if (condition.includes('雪')) return '❄️';
+  if (condition.includes('雷')) return '⛈️';
+  return '🌤️';
+});
+
+// 6. 切换温度单位
+const toggleUnit = () => {
+  isCelsius.value = !isCelsius.value;
+};
+
+// 7. 模拟天气数据查询(实际项目中会调用API)
+const searchWeather = async (city?: string) => {
+  // 清空错误信息
+  errorMessage.value = '';
+  
+  // 确定要查询的城市
+  const targetCity = city || searchQuery.value.trim();
+  if (!targetCity) {
+    errorMessage.value = '请输入城市名称';
+    return;
+  }
+  
+  try {
+    isLoading.value = true;
+    
+    // 模拟API请求延迟
+    await new Promise(resolve => setTimeout(resolve, 800));
+    
+    // 模拟返回的天气数据(实际项目中从API获取)
+    const mockWeatherData: WeatherData = {
+      city: targetCity,
+      temperature: Math.floor(Math.random() * 40) - 10, // -10到30度之间
+      condition: ['晴朗', '多云', '小雨', '中雨', '雷阵雨', '小雪'][Math.floor(Math.random() * 6)],
+      humidity: Math.floor(Math.random() * 50) + 30, // 30%到80%
+      windSpeed: Math.floor(Math.random() * 20) + 1, // 1到20 km/h
+      pressure: Math.floor(Math.random() * 50) + 1000, // 1000到1050 hPa
+      date: new Date()
+    };
+    
+    currentWeather.value = mockWeatherData;
+    
+    // 添加到历史记录(去重)
+    if (!searchHistory.value.includes(targetCity)) {
+      searchHistory.value.unshift(targetCity);
+      // 限制历史记录数量
+      if (searchHistory.value.length > 5) {
+        searchHistory.value.pop();
+      }
+    }
+    
+    // 清空搜索框
+    searchQuery.value = '';
+  } catch (err) {
+    errorMessage.value = '查询天气失败,请重试';
+    console.error('天气查询错误:', err);
+  } finally {
+    isLoading.value = false;
+  }
+};
+
+// 8. 移除历史记录
+const removeHistory = (index: number) => {
+  searchHistory.value.splice(index, 1);
+};
+</script>
+
+<style scoped>
+.weather-app {
+  max-width: 600px;
+  margin: 2rem auto;
+  padding: 0 1rem;
+  font-family: 'Segoe UI', Roboto, Oxygen, Ubuntu, sans-serif;
+}
+
+.search-container {
+  display: flex;
+  gap: 10px;
+  margin: 2rem 0;
+}
+
+.search-container input {
+  flex: 1;
+  padding: 10px;
+  border: 1px solid #e2e8f0;
+  border-radius: 4px;
+  font-size: 1rem;
+}
+
+.search-container button {
+  padding: 10px 20px;
+  background-color: #3182ce;
+  color: white;
+  border: none;
+  border-radius: 4px;
+  cursor: pointer;
+}
+
+.loading, .error {
+  text-align: center;
+  padding: 1rem;
+  border-radius: 4px;
+  margin-bottom: 1rem;
+}
+
+.loading {
+  background-color: #ebf8ff;
+  color: #3182ce;
+}
+
+.error {
+  background-color: #fff5f5;
+  color: #e53e3e;
+}
+
+.weather-card {
+  background-color: white;
+  border-radius: 8px;
+  box-shadow: 0 4px 6px -1px rgba(0, 0, 0, 0.1);
+  padding: 1.5rem;
+  margin-bottom: 2rem;
+}
+
+.weather-header {
+  display: flex;
+  justify-content: space-between;
+  align-items: center;
+  margin-bottom: 1.5rem;
+  padding-bottom: 1rem;
+  border-bottom: 1px solid #edf2f7;
+}
+
+.weather-header h2 {
+  margin: 0;
+  color: #2d3748;
+}
+
+.date {
+  color: #718096;
+}
+
+.weather-main {
+  display: flex;
+  justify-content: space-around;
+  align-items: center;
+  margin-bottom: 1.5rem;
+}
+
+.temperature {
+  display: flex;
+  align-items: baseline;
+}
+
+.temperature .value {
+  font-size: 4rem;
+  font-weight: bold;
+  color: #2d3748;
+}
+
+.temperature .unit {
+  font-size: 1.5rem;
+  color: #4a5568;
+  cursor: pointer;
+  margin-left: 0.5rem;
+  text-decoration: underline;
+}
+
+.condition {
+  text-align: center;
+}
+
+.condition .icon {
+  font-size: 3rem;
+  margin-bottom: 0.5rem;
+}
+
+.condition .text {
+  font-size: 1.2rem;
+  color: #4a5568;
+}
+
+.weather-details {
+  display: grid;
+  grid-template-columns: repeat(3, 1fr);
+  gap: 1rem;
+  padding-top: 1rem;
+  border-top: 1px solid #edf2f7;
+}
+
+.detail {
+  text-align: center;
+}
+
+.detail .label {
+  color: #718096;
+  font-size: 0.9rem;
+}
+
+.detail .value {
+  color: #2d3748;
+  font-weight: 500;
+}
+
+.history-section {
+  margin-top: 2rem;
+}
+
+.history-section h3 {
+  color: #4a5568;
+  margin-bottom: 1rem;
+}
+
+.history-tags {
+  display: flex;
+  flex-wrap: wrap;
+  gap: 0.5rem;
+}
+
+.history-tag {
+  background-color: #edf2f7;
+  padding: 0.5rem 1rem;
+  border-radius: 20px;
+  font-size: 0.9rem;
+  cursor: pointer;
+  display: inline-flex;
+  align-items: center;
+  gap: 0.5rem;
+}
+
+.remove-btn {
+  background: none;
+  border: none;
+  color: #718096;
+  cursor: pointer;
+  font-size: 1rem;
+  padding: 0;
+  line-height: 1;
+}
+
+.remove-btn:hover {
+  color: #e53e3e;
+}
 </style>
+