旅游网站(携程旅行网页学习 vue3+element)

旅游网站

1. 创建项目

  1. 在你要创建项目的路径下打开vscode,新建终端,然后输入vue ui,进入Vue项目管理器。
  2. 选择“创建”,确定项目路径,并点击“在此创建新项目”。
  3. 在项目文件夹中输入项目名称,点击下一步;
  4. 选择“手动配置项目”,然后点击下一步;
  5. 增加选择“router”和“css”两个配置项,然后点击下一步;
  6. 选择 CSS 预处理器。node-sass 是自动编译实时的,dart-sass 需要保存后才会生效,此处建议选择 Sass/SCSS(with dart-sass);pick a linter / formatter config可以选择第一个,只检查错误。选择完成后就可以创建项目。
  7. 提示创建成功后可以用vscode打开新建的项目。

2. 新建首页

注意:App.vue是整个项目的入口文件,这个文件保持干净,后续内容多了,结构才不出错,所以新建一个首页index.vue

  1. 为避免系统提示要组合词命名文件的错误,在package.json文件中修改“rules”的值:

    "rules": {
          "vue/multi-word-component-names": "off"
        }
    
  2. 在src文件夹下的views文件夹中新建index.vue。在index.vue文件中,先设置一个简单页面,输入如下代码:(安装了插件就可以输入vue,然后选择默认vue结构,会自动补全vue结构)

    <template>
      <h1>首页</h1>
    </template>
    
    <script>
    export default {
    
    }
    </script>
    
    <style>
    
    </style>
    
  3. 按照路由的配置方法,在router文件下设置index.js文件,把不需要的路由删除或者注释。两步走,1引入子页面,2设置路由。

    代码如下

    import { createRouter, createWebHashHistory } from 'vue-router'
    // 1.引入
    import Index from '../views/index.vue'
    
    const routes = [
      // 2.设置路由
      {
        path: '/',
        name: 'home',
        component: Index
      } 
    ]
    
    const router = createRouter({
      history: createWebHashHistory(),
      routes
    })
    
    export default router
    

    如果显示index出错
    error Component name "index" should always be multi-word
    解决方式:
    在vue.config.js中添加代码lintOnSave:false
    后重启项目即可

  4. 修改App.vue文件为如下代码,在这个页面中引入需要的页

    <template>
      <router-view/>
    </template>
    

效果图:
在这里插入图片描述

3. 引入ElementPlus组件

  1. 用vscode打开项目,然后在终端中,项目路径下运行命令:npm install element-plus --save

  2. 在main.js文件中配置elementplus和ico图标库,导入方法见element官网,
    https://element-plus.org/zh-CN/,修改main.js为如下代码:

    import { createApp } from 'vue'
    import App from './App.vue'
    import router from './router'
    // ElementPlus 相关库的导入
    import ElementPlus from 'element-plus'
    import 'element-plus/dist/index.css'
    import zhCn from 'element-plus/es/locale/lang/zh-cn'
    // 导入icon图标库
    import * as ElementPlusIconsVue from '@element-plus/icons-vue'
    // 导入暗黑模式主题,在index.html文件中:需在 html 上添加一个名为 dark 的类 
    import 'element-plus/theme-chalk/dark/css-vars.css'
    
    // 创建一个应用
    const app = createApp(App)
    // 注册elementplus
    app.use(ElementPlus, {
      locale: zhCn,
    })
    // 注册elementplus图标
    for (const [key, component] of Object.entries(ElementPlusIconsVue)) {
      app.component(key, component)
    }
    // 注册路由后挂载
    app.use(router).mount('#app')
    

4. 高仿携程旅游页面,制作index.vue

https://hotels.ctrip.com/?allianceid=4897&sid=799747&ouid=xiecheng370&bd_creative=27771225887&bd_vid=11944392447257964181&keywordid=137481675338【携程旅游】

在这里插入图片描述

  1. 首页布局。进入element官网,https://element-plus.org/zh-CN/,选择组件,然后选择Basic 基础组件中的Container 布局容器。根据上面的图,选择左上下结构,复制代码。
    在这里插入图片描述

  2. 粘贴到index.vue中里面并进行修改。此时也可以启动服务器查看布局结果
    代码如下:

    <template>
      <!-- <h1>首页</h1> -->
      <div class="common-layout">
        <el-container>
          <el-aside width="200px" class="test1">Aside</el-aside>
          <el-container>
            <el-header class="test2">Header</el-header>
            <el-main class="test3">Main</el-main>
          </el-container>
        </el-container>
      </div>
    </template>
    
    <script>
    export default {
    
    }
    </script>
    
    <style lang="scss">
    .test1{
        background-color: rgb(177, 208, 255);
        height: 100vh;
    }
    .test2{
        background-color: rgb(110, 160, 235);
    }
    .test3{
        background-color: rgb(187, 239, 255);
    }
    
    </style>
    

    效果图:
    在这里插入图片描述

  3. 将index.vue页面中的三个组成部分拆分成menu.vue、logo.vue和main.vue三个页
    (1)在views文件夹中新建menu.vue、logo.vue和main.vue三个页面,然后将这三个组成部分的代码分别输入到对应的页面中。具体如下:

menu.vue代码:
<template>
   <el-aside width="200px">Aside</el-aside>
</template>

<script>
export default {

}
</script>

<style lang="scss">
.el-aside{
    background-color: rgb(177, 208, 255);
    height: 100vw;
}
</style>

logo.vue代码:
<template>
  <el-header>Header</el-header>
</template>

<script>
export default {

}
</script>

<style lang="scss">

.el-header{
  background-color: rgb(110, 160, 235);
}
</style>

main.vue代码:
<template>
  <el-main>Main</el-main>
</template>

<script>
export default {

}
</script>

<style lang="scss">

.el-main{
  background-color: rgb(187, 239, 255);
}

</style>

(2)修改index.vue,将三个组件页面引入到index.vue中,然后设置组件标签,代码如下:

<template>
  <!-- <h1>首页</h1> -->
  <div class="common-layout">
    <el-container>
      <MenuView></MenuView>
      <el-container>
        <el-header><LogoView></LogoView></el-header>
        <el-main><MainView></MainView></el-main>
      </el-container>
    </el-container>
  </div>
</template>

<script setup>
// 1.script 里加setup,引入要加载的子页面
    import MenuView from './menu.vue'
    import MainView from './main.vue'
    import LogoView from './logo.vue'
</script>

<style lang="scss">


</style>

效果图:
在这里插入图片描述

5. 制作左边菜单栏

功能是可以展开和隐藏菜单栏,所以要在elementplus中找到相对应的组件,想要的效果是munu菜单中的Collapse 折叠面板。
在这里插入图片描述

  1. 打开Collapse 折叠面板的代码,分析其哪些功能代码是我们需要的,不要直接全部复制。leftmenu.vue的代码如下:

    <template>
      <el-aside width="200px">
        <!-- 1.隐藏展开按钮 -->
          <el-radio-group v-model="isCollapse" style="margin-bottom: 20px">
            <el-radio-button :value="false">expand</el-radio-button>
            <el-radio-button :value="true">collapse</el-radio-button>
          </el-radio-group>
          <!-- 2菜单标签 -->
          <el-menu
          default-active="2"
          class="el-menu-vertical-demo"
          :collapse="isCollapse"
          @open="handleOpen"
          @close="handleClose"
        >
          <!-- 3根据页面的布局需要第二组菜单项 -->
          <el-menu-item index="2">
            <el-icon><icon-menu /></el-icon>
            <template #title>酒店</template>
          </el-menu-item>
        </el-menu>
    
      </el-aside>
    </template>
    
    <script setup>
    import { ref } from 'vue'
    // 设置isCollapse的值为true,用于控制展开或隐藏
    const isCollapse = ref(true)
    // 导入icon图标,没用到的组件导入会报错,可以在package.json文件中
    // 增加一条rules:"no-unused-vars": "off"
    import {
      Document,
      Menu as IconMenu,
      Location,
      Setting,
    } from '@element-plus/icons-vue'
    </script>
    
    <style>
    .el-menu-vertical-demo:not(.el-menu--collapse) {
      width: 200px;
      min-height: 400px;
    }
    </style>
    
  2. 启动服务器,查看此时网页效果
    效果图:
    在这里插入图片描述

    在这里插入图片描述

  3. 修改上面的代码,实现与网站几乎相同的效果。

(1)将展开隐藏按钮标签放入到菜单标签,并将展开隐藏的文字提示改为“|||”,并且将两个标签设置v-show属性,展开的时候显示,隐藏的时候隐藏标识符号。

(2)在script里面定义左侧菜单数组

(3)在菜单项中通过v-for去实现显示菜单组

(4)更改icon图标,可以在icon图标库中复制图标名字加到已导入的图标项中

(5)修改菜单项中设置icon图标的代码,实现动态加载图标

leftmenu.vue的代码改动为如下所示:

<template>
    <el-aside width="200px">

        <!-- 1.隐藏展开按钮 -->
        <el-radio-group v-model="isCollapse" style="margin-bottom: 20px">
            <!-- <el-radio-button :value="false">expand</el-radio-button>
            <el-radio-button :value="true">collapse</el-radio-button> -->
            <!-- 6.修改为:将展开隐藏按钮标签放入到菜单标签,
            并将展开隐藏的文字提示改为“|||”,
            并且将两个标签设置v-show属性,展开的时候显示,隐藏的时候隐藏标识符号 -->
            <el-radio-button :value="false" v-show="isCollapse">|||</el-radio-button>
            <el-radio-button :value="true" v-show="!isCollapse">|||</el-radio-button>
        </el-radio-group>
        <!-- 2菜单标签 -->
        <el-menu default-active="2" class="el-menu-vertical-demo" 
        :collapse="isCollapse" @open="handleOpen" @close="handleClose">
        <!-- 3根据页面的布局需要第二组菜单项 -->
            <!-- <el-menu-item index="2">
                <el-icon><icon-menu /></el-icon>
                <template #title>酒店</template>
            </el-menu-item> -->
            <!-- 8.需要多个这个菜单项就用v-for实现-->
            <el-menu-item index="v.id" v-for="v in asidelist" :key="v.id">
                <!-- 11.实现动态icon图标 -->
                <el-icon><component :is="v.icon"></component></el-icon>
                <template #title>{{v.title}}</template>
            </el-menu-item>
        </el-menu>
    </el-aside>
</template>

<script setup>
import { ref } from "vue";

// 设置isCollapse的值为true,用于控制展开或隐藏
const isCollapse = ref(true)
// 导入icon图标,没用到的组件导入会报错,可以在package.json文件中
// 增加一条rules:"no-unused-vars": "off"
import {
    Document,
    Menu as IconMenu,
    Location,
    Setting,
    // 9.增加新的图标
    OfficeBuilding,
    Bowl
} from '@element-plus/icons-vue'

// 7.定义左侧菜单数组,10.增加icon
let asidelist=ref([
    {id:1,title:'酒店',icon:OfficeBuilding},
    {id:2,title:'机票',icon:Document},
    {id:3,title:'火车票',icon:Setting},
    {id:4,title:'旅游',icon:Bowl},
    {id:5,title:'景点',icon:Location},
    {id:6,title:'汽车票',icon:Document},
    {id:7,title:'船票',icon:IconMenu},
    {id:8,title:'门票',icon:OfficeBuilding},
])
</script>

<style lang="scss">
.el-aside{
    background-color: rgb(177, 208, 255);

}
.el-menu-vertical-demo:not(.el-menu--collapse) {
    width: 200px;
    min-height: 400px;
}
</style>
  1. 启动服务器,查看此时的显示效果
    在这里插入图片描述

    在这里插入图片描述

  2. 调整按钮位置

    <template>
        <el-aside width="200px">
    
            
            <!-- 菜单标签 -->
            <el-menu 
                default-active="2" 
                class="el-menu-vertical-demo" 
                :collapse="isCollapse" 
                @open="handleOpen" 
                @close="handleClose"
            >
                <!-- 隐藏展开按钮 -->
                <el-radio-group v-model="isCollapse" style="margin-bottom: 20px">
                    <el-radio-button :value="false" v-show="isCollapse">|||</el-radio-button>
                    <el-radio-button :value="true" v-show="!isCollapse">|||</el-radio-button>
                </el-radio-group>
    
                <!-- 需要多个这个菜单项就用v-for实现-->
                <el-menu-item index="v.id" v-for="v in asidelist" :key="v.id">
                    <!-- 实现动态icon图标 -->
                    <el-icon><component :is="v.icon"></component></el-icon>
                    <template #title>{{v.title}}</template>
                </el-menu-item>
            </el-menu>
        </el-aside>
    </template>
    
    <script setup>
    import { ref } from "vue";
    
    // 设置isCollapse的值为true,用于控制展开或隐藏
    const isCollapse = ref(true)
    // 导入icon图标,没用到的组件导入会报错,可以在package.json文件中
    // 增加一条rules:"no-unused-vars": "off"
    import {
        Document,
        Menu as IconMenu,
        Location,
        Setting,
        // 9.增加新的图标
        OfficeBuilding,
        Bowl
    } from '@element-plus/icons-vue'
    
    // 定义左侧菜单数组,10.增加icon
    let asidelist=ref([
        {id:1,title:'酒店',icon:OfficeBuilding},
        {id:2,title:'机票',icon:Document},
        {id:3,title:'火车票',icon:Setting},
        {id:4,title:'旅游',icon:Bowl},
        {id:5,title:'景点',icon:Location},
        {id:6,title:'汽车票',icon:Document},
        {id:7,title:'船票',icon:IconMenu},
        {id:8,title:'门票',icon:OfficeBuilding},
    ])
    </script>
    
    <style lang="scss">
    .el-aside{
        background-color: rgb(177, 208, 255);
    
    }
    .el-menu-vertical-demo:not(.el-menu--collapse) {
        width: 200px;
        min-height: 400px;
    }
    </style>
    

    效果:
    在这里插入图片描述

    在这里插入图片描述

6. 字体图标的引入

  1. 下载第三方图标,进入阿里巴巴矢量图库https://www.iconfont.cn/?spm=a313x.manage_type_myprojects.i3.2.74bc3a812ze26a,并保存在src中
    在这里插入图片描述

    在这里插入图片描述

  2. 全局引入
    在main.js中加上代码import '@/assets/font/iconfont.css'

  3. 修改menu中代码

    <template>
        <el-aside width="200px">
    
            
            <!-- 菜单标签 -->
            <el-menu 
                default-active="2" 
                class="el-menu-vertical-demo" 
                :collapse="isCollapse" 
                @open="handleOpen" 
                @close="handleClose"
            >
                <!-- 隐藏展开按钮 -->
                <el-radio-group v-model="isCollapse" style="margin-bottom: 20px">
                    <el-radio-button :value="false" v-show="isCollapse">|||</el-radio-button>
                    <el-radio-button :value="true" v-show="!isCollapse">|||</el-radio-button>
                </el-radio-group>
    
                <!-- 需要多个这个菜单项就用v-for实现-->
                <el-menu-item index="v.id" v-for="v in asidelist" :key="v.id">
                    <!-- 实现动态icon图标 -->
                    <!-- <el-icon><component :is="v.icon"></component></el-icon> -->
                    <i class="iconfont" :class="v.icon"></i>
                    <template #title>{{v.title}}</template>
                </el-menu-item>
            </el-menu>
        </el-aside>
    </template>
    
    <script setup>
    import { ref } from "vue";
    
    // 设置isCollapse的值为true,用于控制展开或隐藏
    const isCollapse = ref(true)
    
    // 导入icon图标,没用到的组件导入会报错,可以在package.json文件中
    // 增加一条rules:"no-unused-vars": "off"
    import {
        Document,
        Menu as IconMenu,
        Location,
        Setting,
        // 9.增加新的图标
        OfficeBuilding,
        Bowl
    } from '@element-plus/icons-vue'
    
    // 定义左侧菜单数组,10.增加icon
    // let asidelist=ref([
    //     {id:1,title:'酒店',icon:OfficeBuilding},
    //     {id:2,title:'机票',icon:Document},
    //     {id:3,title:'火车票',icon:Setting},
    //     {id:4,title:'旅游',icon:Bowl},
    //     {id:5,title:'景点',icon:Location},
    //     {id:6,title:'汽车票',icon:Document},
    //     {id:7,title:'船票',icon:IconMenu},
    //     {id:8,title:'门票',icon:OfficeBuilding},
    // ])
    
    // 引入第三方图标
    let asidelist = ref([
        {id:1,title:'酒店',icon:'icon-jiudian'},
        {id:2,title:'机票',icon:'icon-lvxing'},
        {id:3,title:'火车票',icon:'icon-huoche'},
        {id:4,title:'旅游',icon:'icon-lvyou1'},
        {id:5,title:'景点',icon:'icon-jingdian'},
        {id:6,title:'汽车票',icon:'icon-tuijianqiche'},
        {id:7,title:'船票',icon:'icon-lunchuan'},
        {id:8,title:'门票',icon:'icon-menpiao'},
    ])
    </script>
    
    <style lang="scss">
    .el-aside{
        background-color: rgb(177, 208, 255);
    
    }
    .el-menu-vertical-demo:not(.el-menu--collapse) {
        width: 200px;
        min-height: 400px;
    }
    </style>
    
  4. 图标效果
    在这里插入图片描述

7. 左侧导航栏样式

  1. 新建样式index.css

  2. 代码
    index.css

    /* 左侧导航栏 */
    .aside {
        width: 200px;
    }
    .aside-list {
         /* 固定定位 */
        position: fixed;
        left: 0;
        width: 200px;
    }
    .aside .el-menu {
        height:100vh;
    
    }
    /* 字体图标的样式 */
    .aside i {
        font-size: 24px;
        font-weight: 500;
        margin-right: 5px;
    }
    .aside .el-radio-button__inner {
        border:none;
        font-size: 20px;
    }
    

​ menu.vue

<template>
    
    <el-aside width="200px" class="aside">
        <div class="aside-list">
            <!-- 菜单标签 -->
            <el-menu 
                default-active="2" 
                class="el-menu-vertical-demo" 
                :collapse="isCollapse" 
                @open="handleOpen" 
                @close="handleClose"
            >
                <!-- 隐藏展开按钮 -->
                <el-radio-group v-model="isCollapse" style="margin-bottom: 20px">
                    <el-radio-button :value="false" v-show="isCollapse">|||</el-radio-button>
                    <el-radio-button :value="true" v-show="!isCollapse">|||</el-radio-button>
                </el-radio-group>

                <!-- 需要多个这个菜单项就用v-for实现-->
                <el-menu-item index="v.id" v-for="v in asidelist" :key="v.id">
                    <!-- 实现动态icon图标 -->
                    <!-- <el-icon><component :is="v.icon"></component></el-icon> -->
                    <i class="iconfont" :class="v.icon"></i>
                    <template #title>{{v.title}}</template>
                </el-menu-item>
            </el-menu>
        </div>
        
    </el-aside>
</template>

<script setup>
import { ref } from "vue";

// 设置isCollapse的值为true,用于控制展开或隐藏
const isCollapse = ref(true)

// 导入icon图标,没用到的组件导入会报错,可以在package.json文件中
// 增加一条rules:"no-unused-vars": "off"
import {
    Document,
    Menu as IconMenu,
    Location,
    Setting,
    // 9.增加新的图标
    OfficeBuilding,
    Bowl
} from '@element-plus/icons-vue'

// 定义左侧菜单数组,10.增加icon
// let asidelist=ref([
//     {id:1,title:'酒店',icon:OfficeBuilding},
//     {id:2,title:'机票',icon:Document},
//     {id:3,title:'火车票',icon:Setting},
//     {id:4,title:'旅游',icon:Bowl},
//     {id:5,title:'景点',icon:Location},
//     {id:6,title:'汽车票',icon:Document},
//     {id:7,title:'船票',icon:IconMenu},
//     {id:8,title:'门票',icon:OfficeBuilding},
// ])

// 引入第三方图标
let asidelist = ref([
    {id:1,title:'酒店',icon:'icon-jiudian'},
    {id:2,title:'机票',icon:'icon-lvxing'},
    {id:3,title:'火车票',icon:'icon-huoche'},
    {id:4,title:'旅游',icon:'icon-lvyou1'},
    {id:5,title:'景点',icon:'icon-jingdian'},
    {id:6,title:'汽车票',icon:'icon-tuijianqiche'},
    {id:7,title:'船票',icon:'icon-lunchuan'},
    {id:8,title:'门票',icon:'icon-menpiao'},
])
</script>

<style lang="scss">

</style>
```
  1. 引入样式
    在main.js中import '@/assets/css/index.css'

  2. 效果图
    在这里插入图片描述

    遇到问题:

    1. 隐藏张开按钮会跟随移动

    解决方式:

    ​ 修改#appde 样式
    ​ 删除text-align: center

    如不清楚,请看12.位置区域中的解决方式

8. 头部导航栏

  1. 修改logo.vue中的代码
    <template>
      <div>
        <div class="headerNav">
          <!-- logo -->
          <div class="header-logo">logo</div>
          <!-- 搜索框 -->
          <div class="header-search">搜索框</div>
          <!-- 导航菜单 -->
          <div class="header-right">导航菜单</div>
        </div>
      </div>
    </template>
    
    <script>
    export default {
    
    }
    </script>
    
    <style lang="scss">
    
    .el-header{
      background-color: rgb(110, 160, 235);
    }
    </style>
    

​ 效果图:
在这里插入图片描述

  1. 横向排列、固定定位
    修改index.css文件

    /* 页面头部 */
    /* 宽度响应 */
    .el-header {
        min-width: 1280px;
    }
    .headerNav {
        display: flex;
        justify-content: space-between;
        align-items: center;
        /* 水平垂直对齐 */
        height: 70px;
        width:1620px;
        /* 固定定位 */
        position:fixed;
        top:0;
        background-color: #fff;
        z-index: 999;
        padding: 10px 0;
    }
    

    删除logo.vue中定义的css样式

  2. 效果图
    在这里插入图片描述

  3. logo图
    在这里插入图片描述

  4. 修改代码

    <template>
      <div>
        <div class="headerNav">
          <!-- logo -->
          <div class="header-logo">
            <img src="@/assets/images/logo.png" alt="">
          </div>
          <!-- 搜索框 -->
          <div class="header-search">搜索框</div>
          <!-- 导航菜单 -->
          <div class="header-right">导航菜单</div>
        </div>
      </div>
    </template>
    
    <script>
    export default {
    
    }
    </script>
    
    <style lang="scss">
    
    </style>
    
  5. 修改样式

    .headerNav {
        display: flex;
        justify-content: space-between;
        align-items: center;
        /* 水平垂直对齐 */
        height: 70px;
        width:1400px;
        /* 固定定位 */
        position:fixed;
        top:0;
        background-color: #fff;
        z-index: 999;
        padding: 10px 0;
    }
    .headerNav img {
        width: 100%;
        height: 50px;
    }
    

    效果图:
    在这里插入图片描述

9. 搜索框

  1. 官网地址Input 输入框 | Element Plus (element-plus.org)

  2. 修改代码
    logo.vue

    <template>
      <div>
        <div class="headerNav">
          <!-- logo -->
          <div class="header-logo">
            <img src="@/assets/images/logo.png" alt="">
          </div>
          <!-- 搜索框 -->
          <div class="header-search">
            <el-input
                v-model="input3"
                placeholder="请输入内容"
                class="input-with-select"
                >
                <template #append>
                    <el-button :icon="Search" />
                </template>
            </el-input>
          </div>
          <!-- 导航菜单 -->
          <div class="header-right">导航菜单</div>
        </div>
      </div>
    </template>
    
    <script setup>
    //引入搜索图标
    import { Search } from '@element-plus/icons-vue'
    </script>
    
    <style lang="scss">
    
    </style>
    

    index.css

    .header-search {
        flex: 1;
        margin: 0 2rem;
    }
    /* 按钮的颜色  */
    .headerNav .el-input-group__append {
        background-color: var(--el-color-primary);
        color: #fff;
    }
    .heaer-right .el-link {
        margin: 0 20px;
    }
    

    效果图:
    在这里插入图片描述

10. 头部下拉菜单

  1. 官网Link 链接 | Element Plus (element-plus.org)

  2. 修改代码
    logo.vue

    <template>
      <div>
        <div class="headerNav">
          <!-- logo -->
          <div class="header-logo">
            <img src="@/assets/images/logo.png" alt="">
          </div>
          <!-- 搜索框 -->
          <div class="header-search">
            <el-input
                v-model="input3"
                placeholder="请输入内容"
                class="input-with-select"
                >
                <template #append>
                    <el-button :icon="Search" />
                </template>
            </el-input>
          </div>
          <!-- 导航菜单 -->
          <div class="header-right">
            <el-link :icon="Avatar">请登录</el-link>
            <el-link target="_blank">注册</el-link>
            <el-link>
              <el-dropdown>
                  <span class="el-dropdown-link">
                  我的订单
                  <el-icon class="el-icon--right">
                      <arrow-down />
                  </el-icon>
                  </span>
                  <template #dropdown>
                    <el-dropdown-menu>
                        <el-dropdown-item v-for="v in Myorder" :key="v.id">{{v.title}}</el-dropdown-item>
                    </el-dropdown-menu>
                  </template>
              </el-dropdown>
            </el-link>
          </div>
        </div>
      </div>
    </template>
    
    <script setup>
    //引入搜索图标
    import { Search,Avatar,ArrowDown } from '@element-plus/icons-vue'
    import { ref } from 'vue'
    //下拉列表
    let Myorder = ref([
        {id:1,title:'酒店订单'},
        {id:2,title:'火车票订单'},
        {id:3,title:'飞机票订单'},
        {id:4,title:'旅游订单'},
        {id:5,title:'全部订单'}
    ])
    </script>
    
    <style lang="scss">
    
    </style>
    

    index.css

    .header-right .el-link {
        margin: 0 20px;
    }
    /* 伪类选择器:可以用来选取获得焦点的元素
    当鼠标移入上去后取消黑色边框 */
    .el-dropdown-link:focus{
       outline: none;
    }
    

    效果图:
    在这里插入图片描述

11. 区域模块

  1. 修改代码
    main.vue

    <template>
      <div class="searchlist">
        <div class="searchlist-item">1</div>
        <div class="searchlist-item">2</div>
        <div class="searchlist-item">3</div>
        <div class="searchlist-item">4</div>
        <div class="searchlist-item">5</div>
      </div>
    </template>
    
    <script>
    export default {
    
    }
    </script>
    
    <style lang="scss">
    
    </style>
    

    index.css

    /* 区域展示部分 */
    .searchlist {
        width: 1400px;
        display: grid;
        grid-template-columns: repeat(5,auto);
        margin-top: 15px;
    }
    .searchlist-item {
        border: 1px solid #dadada;
        margin-left: -1px;
    }
    

    效果图:
    在这里插入图片描述

  2. 内部搜索框、时间选择器

    遇到问题

    输入框一直有黑边

    解决方式

    css中增加input样式

    main.vue

    <template>
      <div class="searchlist">
        <div class="searchlist-item">
            <div class="search-box">
              <!-- 使用for和id进行表单绑定 -->
                <label for="hotels-destination">目的地/酒店名称</label>
                <input class="input" type="text" id="hotels-destination" placeholder="北京">
            </div>
        </div>
        <div class="searchlist-item">
            <div class="search-box">
                <label>入住时间/退房时间</label>
                <el-date-picker
                    v-model="value1"
                    type="datetimerange"
                    start-placeholder="Start Date"
                    end-placeholder="End Date"
                    :default-time="defaultTime1"
                    />
            </div>
        </div>
        <div class="searchlist-item">
              <div class="search-box">
                <label for="room-guest">房间及住客</label>
                <input class="input" type="text" id="room-guest" placeholder="1间/1位">
            </div>
        </div>
        <div class="searchlist-item">
              <div class="search-box">
                <label for="keyword">关健词(选填)</label>
                <input class="input" type="text" id="keyword" placeholder="火车/酒店名称或区域">
            </div>
        </div>
        <div class="searchlist-item">
            <el-button type="primary" :icon="Search"></el-button>
        </div>
      </div>
    </template>
    
    <script setup>
    import {ref} from 'vue'
    import {Search} from '@element-plus/icons-vue'
    </script>
    
    <style lang="scss">
    
    </style>
    

    index.css

    /* 右侧下面展示 */
    .el-main {
        min-width: 1280px;
        overflow: hidden;
    }
    /* 目的地/酒店名称 */
    .search-box {
        padding: 8px 12px;
    }
    .search-box label {
        color: #999;
        display: block;
    }
    .search-box input {
        border:none;
        font-size: 16px;
        font-weight: 700;
        padding-top: 15px;
    }
    
    .input{
        outline: none;
    }
    /* 伪类选择器,当元素获得焦点的时候,能够控制父元素的样式 */
    .searchlist-item:focus-within {
        border-bottom: 2px solid var(--el-color-primary);
    }
    /* 入住时间/退房时间 取消边框的颜色 */
    .search-box .el-date-editor.el-input__wrapper {
        box-shadow:none;
    }
    .search-box .el-date-editor.el-input__wrapper:hover {
        box-shadow:none;
    }
    /* 搜索按钮的样式 */
    .searchlist-item .el-button {
        width: 100%;
        height:100%;
        border-radius: 0;
    }
    .searchlist-item .el-button .el-icon {
        font-size: 30px;
    }
    
    

    效果图:
    在这里插入图片描述

![在这里插入图片描述](https://img-blog.csdnimg.cn/direct/b7e8454de1064819a38ccb10405dfa83.png)

12. 位置区域

使用网格布局

  1. 新建页面,并在index.vue中加入页面

    <template>
      <!-- <h1>首页</h1> -->
     
      <div class="common-layout">
        <el-container>
          <MenuView></MenuView>
          <el-container>
            <el-header><LogoView></LogoView></el-header>
            <el-main><MainView></MainView></el-main>
            <el-top><Toplist></Toplist></el-top>
          </el-container>
        </el-container>
      </div>
    </template>
    
    <script setup>
        // 1.script 里加setup,引入要加载的子页面
        import MenuView from './menu.vue'
        import MainView from './main.vue'
        import LogoView from './logo.vue'
        import Toplist from './toplist.vue';
    </script>
    
    <style lang="scss">
    
    
    </style>
    
  2. 修改代码
    toplist.vue

    <template>
      <div class="topfilter">
        <div>
          <h4>位置区域</h4>
        </div>
        <!-- tabs切换 -->
        <div class="topfilter-tab">
          <el-tabs v-model="activeName" class="demo-tabs" @tab-click="handleClick">
            <!-- 数据绑定 -->
            <el-tab-pane :label="v.label" :name="v.name" v-for="v in topfilterlist" :key="v.id">
              <el-button  v-for="(k,i) in v.list" :key="i" size="small" type="info" plain>{{k}}</el-button>
            </el-tab-pane>
          </el-tabs>
        </div>
        <div>
          <h4>星级价格</h4>
        </div>
        <div class="topfilter-price">
            <el-button size="small" type="info" plain v-for="(v,i) in pricelist" :key="i">{{v}}</el-button>
        </div>
        <div>
          <h4>高级筛选</h4>
        </div>
        <div class="topfilter-filter">
            <el-dropdown v-for="(v,i) in filterlist" :key="i">
                <span class="el-dropdown-link">
                  {{v.title}}
                  <el-icon class="el-icon--right">
                    <!-- 箭头按钮 -->
                    <arrow-down />
                  </el-icon>
                </span>
                <template #dropdown>
                  <el-dropdown-menu>
                    <el-dropdown-item v-for="(k,i) in v.list" :key="i">{{k}}</el-dropdown-item>
                  </el-dropdown-menu>
                </template>
          </el-dropdown>
        </div>
      </div>
    </template>
    
    <script setup>
    import {ref} from 'vue'
    //箭头图标
    import { ArrowDown } from '@element-plus/icons-vue'
    const activeName = ref('first')
    let topfilterlist = ref([
      {id:1,name:'first',label:'热门筛选',list:['天安门广场','王府井','北京大家','天安门广场','王府井','北京大家','天安门广场','王府井','北京大家','天安门广场','王府井','北京大家','天安门广场','王府井','北京大家','天安门广场','王府井','北京大家']},
      {id:2,name:'second',label:'机场车站',list:['北京南站','北京西站','北京火车站','北京南站','北京西站','北京火车站','北京南站','北京西站','北京火车站']},
      {id:3,name:'third',label:'商业区',list:['三里屯','国贸地区','西直门','三里屯','国贸地区','西直门','三里屯','国贸地区','西直门']},
      {id:4,name:'fourth',label:'景点',list:['什刹海','前门大街','五道口','什刹海','前门大街','五道口','什刹海','前门大街','五道口']}
    ])
    let pricelist = ref(['五星(钻级)','四星(钻级)','三星(钻级)','两星(钻级)','一星(钻级)'])
    let filterlist = ref([
      {title:'早餐',list:['含早餐','单份早餐','多份早餐']},
      {title:'支付方式',list:['在线付款','到店付款','闪住']},
      {title:'房型',list:['双床房','三床房','单人床房','大床房','特大床房']},
      {title:'酒店设施',list:['免费停车','室内游泳','单人床房','大床房','特大床房']},
      {title:'住宿类型',list:['免费停车','室内游泳','单人床房','大床房','特大床房']},
      {title:'酒特色主题',list:['免费停车','室内游泳','单人床房','大床房','特大床房']},
    ])
    </script>
    
    <style>
    
    </style>
    

遇到问题:

按钮居中对齐,不靠左

解决方式:

修改#app中的样式(如果在之前隐藏展开框中改过则不会出现问题)
在App.vue中,注释掉text-align: center;

<template>
  
  <router-view/>
</template>

<style lang="scss">
#app {
  font-family: Avenir, Helvetica, Arial, sans-serif;
  -webkit-font-smoothing: antialiased;
  -moz-osx-font-smoothing: grayscale;
  // text-align: center;
  color: #2c3e50;
}

nav {
  padding: 30px;

  a {
    font-weight: bold;
    color: #2c3e50;

    &.router-link-exact-active {
      color: #42b983;
    }
  }
}
</style>

为防止应样式继承产生的问题,将App.vue中的样式都注销

遇到问题:

星级价格中的按钮以及高级筛选的下拉菜单不垂直居中

解决方式:

在index.css中增加样式

.topfilter-price{
    display: flex;
    align-items: center;
}
.topfilter-filter{
    display: flex;
    align-items: center;
}
  1. indext.css(根据页面大小,修改了其他样式)

    /* 左侧导航栏 */
    .aside {
        width: 200px;
    }
    .aside-list {
        /* 固定定位 */
        position: fixed;
        left: 0;
        width: 200px;
    }
    .aside .el-menu {
        height:100vh;
    
    }
    /* 字体图标的样式 */
    .aside i {
        font-size: 24px;
        font-weight: 500;
        margin-right: 5px;
    }
    .aside .el-radio-button__inner {
        border:none;
        font-size: 20px;
    }
    
    
    /* 页面头部 */
    /* 宽度响应 */
    .el-header {
        min-width: 1280px;
    }
    .headerNav {
        display: flex;
        justify-content: space-between;
        align-items: center;
        /* 水平垂直对齐 */
        height: 70px;
        width:1300px;
        /* 固定定位 */
        position:fixed;
        top:0;
        background-color: #fff;
        z-index: 999;
        padding: 10px 0;
    }
    .headerNav img {
        width: 100%;
        height: 50px;
    }
    .header-search {
        flex: 1;
        margin: 0 2rem;
    }
    /* 按钮的颜色  */
    .headerNav .el-input-group__append {
        background-color: var(--el-color-primary);
        color: #fff;
    }
    .header-right .el-link {
        margin: 0 20px;
    }
    /* 伪类选择器:可以用来选取获得焦点的元素
    当鼠标移入上去后取消黑色边框 */
    .el-dropdown-link:focus{
       outline: none;
    }
    /* 右侧下面展示 */
    .el-main {
        min-width: 1280px;
        overflow: hidden;
    }
    /* 区域展示部分 */
    .searchlist {
        width: 1300px;
        display: grid;
        grid-template-columns: repeat(5,auto);
        margin-top: 15px;
    }
    .searchlist-item {
        border: 1px solid #dadada;
        margin-left: -1px;
    }
    /* 目的地/酒店名称 */
    .search-box {
        padding: 8px 12px;
    }
    .search-box label {
        color: #999;
        display: block;
    }
    .search-box input {
        border:none;
        font-size: 16px;
        font-weight: 700;
        padding-top: 15px;
    }
    
    .input{
        outline: none;
    }
    /* 伪类选择器,当元素获得焦点的时候,能够控制父元素的样式 */
    .searchlist-item:focus-within {
        border-bottom: 2px solid var(--el-color-primary);
    }
    /* 入住时间/退房时间 取消边框的颜色 */
    .search-box .el-date-editor.el-input__wrapper {
        box-shadow:none;
    }
    .search-box .el-date-editor.el-input__wrapper:hover {
        box-shadow:none;
    }
    /* 搜索按钮的样式 */
    .searchlist-item .el-button {
        width: 100%;
        height:100%;
        border-radius: 0;
    }
    .searchlist-item .el-button .el-icon {
        font-size: 30px;
    }
    
    /* topfilter开始  */
    .topfilter {
        width: 1300px;
        display:grid;
        grid-template-columns: 130px 1fr;
        /* 列 */
        grid-template-rows:repeat(3,auto);
        /* 3行 */
    }
    .topfilter>div {
        border-bottom:1px solid #ddd;
        padding: 0.2rem 1rem;
    }
    .topfilter h4 {
        line-height: 17px;
        text-align: center;
    }
    .topfilter .el-button:hover {
        background-color: var(--el-color-primary);
        border: 1px solid var(--el-color-primary);
    }
    
    .topfilter-price{
        display: flex;
        align-items: center;
    }
    
    .topfilter-filter{
        display: flex;
        align-items: center;
    }
    
    /* 按钮的间距: */
    .topfilter .el-button {
        margin: 7px;
    }
    /* 高级筛选 */
    .topfilter-filter .el-dropdown {
        margin: 10px 15px;
        /* 下拉菜单间距 */
    }
    
    
  2. 效果图
    在这里插入图片描述

13. 走马灯和列表

  1. 新建文件,并引入
    recommend.vue

    <template>
      <div class="recommend">
        <div class="recommend-left">推荐酒店</div>
        <div class="recommend-left">轮播图</div>
      </div>
    </template>
    
    <script setup>
    import { ref } from "vue";
    
    </script>
    
    <style>
    
    </style>
    

    index.vue

    <template>
      <!-- <h1>首页</h1> -->
     
      <div class="common-layout">
        <el-container>
          <MenuView></MenuView>
          <el-container>
            <el-header><LogoView></LogoView></el-header>
            <el-main>
              <MainView></MainView>
              <Toplist></Toplist>
              <Recommend></Recommend>
            </el-main>
          </el-container>
        </el-container>
      </div>
    </template>
    
    <script setup>
        // 1.script 里加setup,引入要加载的子页面
        import MenuView from './menu.vue'
        import MainView from './main.vue'
        import LogoView from './logo.vue'
        import Toplist from './toplist.vue'
        import Recommend from './recommend.vue'
    </script>
    
    <style lang="scss">
    
    
    </style>
    
  2. 效果图
    在这里插入图片描述

  3. 修改代码,增加视图Recommend-left和Banner
    recommend.vue

<template>
  <div class="recommend">
    <div class="recommend-left">
        <h2>推荐酒店</h2>
        <Recommendlist></Recommendlist>
    </div>
    <div class="recommend-right">
        <Banner></Banner>
    </div>
  </div>
</template>

<script setup>
import { ref } from "vue";

</script>

<style>

</style>

index.css

/* 推荐酒店 */
.recommend {
    width:1300px;
    display:flex;
    gap: 0 20px;
    padding: 10px 0;
}
.recommend-left {
    width: 900px;
}
.recommend-right {
    width: 400px;
}
  1. 修改代码
    banner.vue
<template>
<div class="banner">
    <el-carousel height="620px">
        <el-carousel-item v-for="item in bannerlist" :key="item.id">
           <img :src="`/src/assets/images/${item.img_url}`" alt="">
        </el-carousel-item>
    </el-carousel>
</div>
</template>

<script setup>
import {ref} from 'vue'

let bannerlist = ref([
    {id:1,img_url:'banner1.jpg'},
    {id:2,img_url:'banner2.jpg'},
    {id:3,img_url:'banner3.jpg'},
    {id:4,img_url:'banner4.jpg'},
    {id:5,img_url:'banner5.jpg'},
])
</script>

<style>

</style>

index.css

.banner {
    width: 400px;
    height: 620px;
}
/* 轮播图 */
.banner img {
    width: 100%;
    height: 100%;
    object-fit: cover;
}

遇到问题:

banner.vue中图片不显示

找不到路径

解决方式:

<!--相对路径引用 webpack 帮我们解析时需要使用 require() 把图片路径包裹起来 -->
<!-- 在public文件夹下是绝对路径,不需要经过webpack -->

<img :src="require(`/src/assets/images/${item.img_url}`)" alt="">

5.完善代码
recommend.vue

<template>
  <div class="recommend">
    <div class="recommend-left">
        <h2>推荐酒店</h2>
        <Recommendlist></Recommendlist>
    </div>
    <div class="recommend-right">
        <Banner></Banner>
    </div>
  </div>
</template>

<script setup>
import { ref } from "vue";
import Banner from './banner.vue'
import Recommendlist from './recommend-list.vue'
</script>

<style>

</style>

recommend-list.vue

<template>
<div class="recommend-list">
    <ul>
        <li v-for="v in recommendlist" :key="v.id"> 
            <div class="list-left">
                <div class="list-left-img">
                    <img :src="require(`/src/assets/images/${v.img_url}`)" alt="">
                </div>
                <div class="list-left-title">
                    <div class="list-left-tagandtitle">
                        <h4>{{v.name}}</h4>
                        <img :src="require(`/src/assets/images/${v.starimg_url}`)" alt="">
                        <span class="badge-thumb"></span>
                    </div>
                    <div class="list-left-transport">
                        <p>{{v.transport}}</p>
                    </div>
                    <div class="list-left-tag">
                        <el-tag class="ml-2" type="danger" effect="plain" v-for="(k,i) in v.tag" :key="i">{{k}}</el-tag>
                    </div>
                </div>
            </div>
            <!-- 右侧 -->
            <div class="list-right">
                <div class="list-right-comment">
                    <div class="list-right-describe">
                        <div class="p1">好</div>
                
                        <a href="#">3333条点评</a>
                    </div>
                    <div class="list-right-score">
                        <p>4.5</p>
                    </div>
                </div>
                <div class="list-right-price">
                    <p>¥3333</p>
                     <el-button type="primary">查看详情</el-button>
                </div>
            </div>
        </li>
    </ul>
</div>
</template>

<script setup>
import {ref} from 'vue'

let recommendlist = ref([
    {id:1,name:'北京首都宾馆',img_url:'1.jpg',transport:'天安门/王府井 | 距市中心直线310米',starimg_url:'star5.png',tag:['网球场','商务中心','会议厅']},
    {id:2,name:'北京京伦饭店',img_url:'2.jpg',transport:'国贸CBD | 距市中心直线4.1公里',starimg_url:'star4.png',tag:['停车场','健身室','SPA']},
    {id:3,name:'北京新世界中心同派酒店',img_url:'3.jpg',transport:'前门/天坛公园/崇文门 | 距市中心直线1公里',starimg_url:'star4.png',tag:['桌球室','闪住','酒吧','行李寄存']},
    {id:4,name:'北京新世界中心同派酒店',img_url:'4.jpg',transport:'前门/天坛公园/崇文门 | 距市中心直线1公里',starimg_url:'star4.png',tag:['桌球室','闪住','酒吧','行李寄存']},
])
</script>

<style>

</style>

banner.vue

<template>
<div class="banner">
    <el-carousel height="620px">
        <el-carousel-item v-for="item in bannerlist" :key="item.id">
            <!--相对路径引用 webpack 帮我们解析时需要使用 require() 把图片路径包裹起来 -->
            <!-- 在public文件夹下是绝对路径,不需要经过webpack -->
            <!-- <img :src="require(`../assets/${item.img_url}`)" alt=""> -->
            <img :src="require(`/src/assets/images/${item.img_url}`)" alt="">
        </el-carousel-item>
    </el-carousel>
</div>
</template>

<script setup>
import {ref} from 'vue'

let bannerlist = ref([
    {id:1,img_url:'banner1.jpg'},
    {id:2,img_url:'banner2.jpg'},
    {id:3,img_url:'banner3.jpg'},
    {id:4,img_url:'banner4.jpg'},
    {id:5,img_url:'banner5.jpg'},
])
</script>

<style>

</style>

index.css

/* 左侧导航栏 */
.aside {
    width: 200px;
}
.aside-list {
    /* 固定定位 */
    position: fixed;
    left: 0;
    width: 200px;
}
.aside .el-menu {
    height:100vh;

}
/* 字体图标的样式 */
.aside i {
    font-size: 24px;
    font-weight: 500;
    margin-right: 5px;
}
.aside .el-radio-button__inner {
    border:none;
    font-size: 20px;
}


/* 页面头部 */
/* 宽度响应 */
.el-header {
    min-width: 1280px;
}
.headerNav {
    display: flex;
    justify-content: space-between;
    align-items: center;
    /* 水平垂直对齐 */
    height: 70px;
    width:1300px;
    /* 固定定位 */
    position:fixed;
    top:0;
    background-color: #fff;
    z-index: 999;
    padding: 10px 0;
}
.headerNav img {
    width: 100%;
    height: 50px;
}
.header-search {
    flex: 1;
    margin: 0 2rem;
}
/* 按钮的颜色  */
.headerNav .el-input-group__append {
    background-color: var(--el-color-primary);
    color: #fff;
}
.header-right .el-link {
    margin: 0 20px;
}
/* 伪类选择器:可以用来选取获得焦点的元素
当鼠标移入上去后取消黑色边框 */
.el-dropdown-link:focus{
   outline: none;
}
/* 右侧下面展示 */
.el-main {
    min-width: 1280px;
    overflow: hidden;
}
/* 区域展示部分 */
.searchlist {
    width: 1300px;
    display: grid;
    grid-template-columns: repeat(5,auto);
    margin-top: 15px;
}
.searchlist-item {
    border: 1px solid #dadada;
    margin-left: -1px;
}
/* 目的地/酒店名称 */
.search-box {
    padding: 8px 12px;
}
.search-box label {
    color: #999;
    display: block;
}
.search-box input {
    border:none;
    font-size: 16px;
    font-weight: 700;
    padding-top: 15px;
}

.input{
    outline: none;
}
/* 伪类选择器,当元素获得焦点的时候,能够控制父元素的样式 */
.searchlist-item:focus-within {
    border-bottom: 2px solid var(--el-color-primary);
}
/* 入住时间/退房时间 取消边框的颜色 */
.search-box .el-date-editor.el-input__wrapper {
    box-shadow:none;
}
.search-box .el-date-editor.el-input__wrapper:hover {
    box-shadow:none;
}
/* 搜索按钮的样式 */
.searchlist-item .el-button {
    width: 100%;
    height:100%;
    border-radius: 0;
}
.searchlist-item .el-button .el-icon {
    font-size: 30px;
}

/* topfilter开始  */
.topfilter {
    width: 1300px;
    display:grid;
    grid-template-columns: 130px 1fr;
    /* 列 */
    grid-template-rows:repeat(3,auto);
    /* 3行 */
}
.topfilter>div {
    border-bottom:1px solid #ddd;
    padding: 0.2rem 1rem;
}
.topfilter h4 {
    line-height: 17px;
    text-align: center;
}
.topfilter .el-button:hover {
    background-color: var(--el-color-primary);
    border: 1px solid var(--el-color-primary);
}

.topfilter-price{
    display: flex;
    align-items: center;
}

.topfilter-filter{
    display: flex;
    align-items: center;
}

/* 按钮的间距: */
.topfilter .el-button {
    margin: 7px;
}
/* 高级筛选 */
.topfilter-filter .el-dropdown {
    margin: 10px 15px;
    /* 下拉菜单间距 */
}

/* 推荐酒店 */
.recommend {
    width:1300px;
    display:flex;
    gap: 0 20px;
    padding: 10px 0;
}
.recommend-left {
    width: 880px;
}
.recommend-right {
    width: 420px;
}
.banner {
    width: 420px;
    height: 650px;
    margin-top: 57px;
}

h2{
    font-size: 1.2em;
    margin-left: 30px;
}

/* 轮播图 */
.banner img {
    width: 100%;
    height: 100%;
    object-fit: cover;
}
.recommend-list ul li {
    display: flex;
    background-color: #F4F6F9;
    padding: 10px;
    margin-bottom: 10px;
    height: 127px;
}
.list-left {
    /* 设置了弹性项目如何增大或缩小以适应其弹性容器中可用的空间 */
    flex: 2;
    display: flex;
}
.list-right {
    flex: 1;
    border-left: 1px solid #ddd;
    text-align: right;
    display: flex;
    flex-direction: column;
    justify-content: space-between;
}       
/* 图片的大小 */
.list-left-img {
    flex: 0.7;
}
.list-left-img img {
    width: 100%;
}
.list-left-title {
    flex: 3;
    margin-left: 30px;
}
/* 标题 */
.list-left-tagandtitle {
    font-size: 16px;
    line-height: 3px;
    display: flex;
    margin-bottom: 8px;
    align-items: center;
}
.list-left-tagandtitle img {
    height: 20px;
    width: auto;
    margin: 0 10px;
}
.badge-thumb {
    background:url(@/assets/images/help.png) no-repeat;
    width: 16px;
    height: 16px;
}
.list-left-transport {
    margin-bottom: 8px;
}
.list-left-tag .el-tag {
    margin: 3px;
}
.list-right-comment {
    display:flex;
    justify-content: flex-end;
    margin-bottom: 10px;
    height: 40px;
}
.list-right-describe{
    height: 60px;
}
.p1 {
    color: var(--el-color-primary);
    font-size:16px;
    font-weight: 700;
    align-items: center;
    margin-top: 0px;
    
}
.list-right-describe a {
    color: #999;
    font-size: 11px;
    margin-top: 0px;
}
.list-right-score {
    background-color: var(--el-color-primary);
    color: #fff;
    border-radius: 4px;
    padding: 3px;
    font-size: 18px;
    font-weight: 800;
    margin-left: 9px;
    display: flex;
    align-items: center;
}
.list-right-price p {
    color: var(--el-color-primary);
    font-size:20px;
    font-weight: 700;
    margin-top: 0px;
    margin-bottom: 9px;
}

样式根据电脑屏幕进行了修改

  1. 效果图
    在这里插入图片描述

    在这里插入图片描述

代码整合

1. index.vue

<template>
  <!-- <h1>首页</h1> -->
  <div class="common-layout">
    <el-container>
      <MenuView></MenuView>
      <el-container>
        <el-header><LogoView></LogoView></el-header>
        <el-main>
          <MainView></MainView>
          <Toplist></Toplist>
          <Recommend></Recommend>
        </el-main>
      </el-container>
    </el-container>
  </div>
</template>

<script setup>
    // 1.script 里加setup,引入要加载的子页面
    import MenuView from './menu.vue'
    import MainView from './main.vue'
    import LogoView from './logo.vue'
    import Toplist from './toplist.vue'
    import Recommend from './recommend.vue'
</script>

<style lang="scss">


</style>

2. banner.vue

<template>
<div class="banner">
    <el-carousel height="620px">
        <el-carousel-item v-for="item in bannerlist" :key="item.id">
            <!--相对路径引用 webpack 帮我们解析时需要使用 require() 把图片路径包裹起来 -->
            <!-- 在public文件夹下是绝对路径,不需要经过webpack -->
            <!-- <img :src="require(`../assets/${item.img_url}`)" alt=""> -->
            <img :src="require(`/src/assets/images/${item.img_url}`)" alt="">
        </el-carousel-item>
    </el-carousel>
</div>
</template>

<script setup>
import {ref} from 'vue'

let bannerlist = ref([
    {id:1,img_url:'banner1.jpg'},
    {id:2,img_url:'banner2.jpg'},
    {id:3,img_url:'banner3.jpg'},
    {id:4,img_url:'banner4.jpg'},
    {id:5,img_url:'banner5.jpg'},
])
</script>

<style>

</style>

3. logo.vue

<template>
  <div>
    <div class="headerNav">
      <!-- logo -->
      <div class="header-logo">
        <img src="@/assets/images/logo.png" alt="">
      </div>
      <!-- 搜索框 -->
      <div class="header-search">
        <el-input
            v-model="input3"
            placeholder="请输入内容"
            class="input-with-select"
            >
            <template #append>
                <el-button :icon="Search" />
            </template>
        </el-input>
      </div>
      <!-- 导航菜单 -->
      <div class="header-right">
        <el-link :icon="Avatar">请登录</el-link>
        <el-link target="_blank">注册</el-link>
        <el-link>
          <el-dropdown>
              <span class="el-dropdown-link">
              我的订单
              <el-icon class="el-icon--right">
                  <arrow-down />
              </el-icon>
              </span>
              <template #dropdown>
                <el-dropdown-menu>
                    <el-dropdown-item v-for="v in Myorder" :key="v.id">{{v.title}}</el-dropdown-item>
                </el-dropdown-menu>
              </template>
          </el-dropdown>
        </el-link>
      </div>
    </div>
  </div>
</template>

<script setup>
//引入搜索图标
import { Search,Avatar,ArrowDown } from '@element-plus/icons-vue'
import { ref } from 'vue'
//下拉列表
let Myorder = ref([
    {id:1,title:'酒店订单'},
    {id:2,title:'火车票订单'},
    {id:3,title:'飞机票订单'},
    {id:4,title:'旅游订单'},
    {id:5,title:'全部订单'}
])
</script>

<style lang="scss">

</style>

3. main.vue

<template>
  <div class="searchlist">
    <div class="searchlist-item">
        <div class="search-box">
          <!-- 使用for和id进行表单绑定 -->
            <label for="hotels-destination">目的地/酒店名称</label>
            <input class="input" type="text" id="hotels-destination" placeholder="北京">
        </div>
    </div>
    <div class="searchlist-item">
        <div class="search-box">
            <label>入住时间/退房时间</label>
            <el-date-picker
                v-model="value1"
                type="datetimerange"
                start-placeholder="Start Date"
                end-placeholder="End Date"
                :default-time="defaultTime1"
                />
        </div>
    </div>
    <div class="searchlist-item">
          <div class="search-box">
            <label for="room-guest">房间及住客</label>
            <input class="input" type="text" id="room-guest" placeholder="1间/1位">
        </div>
    </div>
    <div class="searchlist-item">
          <div class="search-box">
            <label for="keyword">关健词(选填)</label>
            <input class="input" type="text" id="keyword" placeholder="火车/酒店名称或区域">
        </div>
    </div>
    <div class="searchlist-item">
        <el-button type="primary" :icon="Search"></el-button>
    </div>
  </div>
</template>

<script setup>
import {ref} from 'vue'
import {Search} from '@element-plus/icons-vue'
</script>

<style lang="scss">

</style>

4. menu.vue

<template>
    
    <el-aside width="200px" class="aside">
        <div class="aside-list">
            <!-- 菜单标签 -->
            <el-menu 
                default-active="2" 
                class="el-menu-vertical-demo" 
                :collapse="isCollapse" 
                @open="handleOpen" 
                @close="handleClose"
            >
                <!-- 隐藏展开按钮 -->
                <el-radio-group v-model="isCollapse" style="margin-bottom: 20px" >
                    <el-radio-button :value="false" v-show="isCollapse">|||</el-radio-button>
                    <el-radio-button :value="true" v-show="!isCollapse">|||</el-radio-button>
                </el-radio-group>

                <!-- 需要多个这个菜单项就用v-for实现-->
                <el-menu-item index="v.id" v-for="v in asidelist" :key="v.id">
                    <!-- 实现动态icon图标 -->
                    <!-- <el-icon><component :is="v.icon"></component></el-icon> -->
                    <i class="iconfont" :class="v.icon"></i>
                    <template #title>{{v.title}}</template>
                </el-menu-item>
            </el-menu>
        </div>
        
    </el-aside>
</template>

<script setup>
import { ref } from "vue";

// 设置isCollapse的值为true,用于控制展开或隐藏
const isCollapse = ref(true)

// 导入icon图标,没用到的组件导入会报错,可以在package.json文件中
// 增加一条rules:"no-unused-vars": "off"
import {
    Document,
    Menu as IconMenu,
    Location,
    Setting,
    // 9.增加新的图标
    OfficeBuilding,
    Bowl
} from '@element-plus/icons-vue'

// 定义左侧菜单数组,10.增加icon
// let asidelist=ref([
//     {id:1,title:'酒店',icon:OfficeBuilding},
//     {id:2,title:'机票',icon:Document},
//     {id:3,title:'火车票',icon:Setting},
//     {id:4,title:'旅游',icon:Bowl},
//     {id:5,title:'景点',icon:Location},
//     {id:6,title:'汽车票',icon:Document},
//     {id:7,title:'船票',icon:IconMenu},
//     {id:8,title:'门票',icon:OfficeBuilding},
// ])

// 引入第三方图标
let asidelist = ref([
    {id:1,title:'酒店',icon:'icon-jiudian'},
    {id:2,title:'机票',icon:'icon-lvxing'},
    {id:3,title:'火车票',icon:'icon-huoche'},
    {id:4,title:'旅游',icon:'icon-lvyou1'},
    {id:5,title:'景点',icon:'icon-jingdian'},
    {id:6,title:'汽车票',icon:'icon-tuijianqiche'},
    {id:7,title:'船票',icon:'icon-lunchuan'},
    {id:8,title:'门票',icon:'icon-menpiao'},
])
</script>

<style lang="scss">

</style>

5. recommend-list.vue

<template>
<div class="recommend-list">
    <ul>
        <li v-for="v in recommendlist" :key="v.id"> 
            <div class="list-left">
                <div class="list-left-img">
                    <img :src="require(`/src/assets/images/${v.img_url}`)" alt="">
                </div>
                <div class="list-left-title">
                    <div class="list-left-tagandtitle">
                        <h4>{{v.name}}</h4>
                        <img :src="require(`/src/assets/images/${v.starimg_url}`)" alt="">
                        <span class="badge-thumb"></span>
                    </div>
                    <div class="list-left-transport">
                        <p>{{v.transport}}</p>
                    </div>
                    <div class="list-left-tag">
                        <el-tag class="ml-2" type="danger" effect="plain" v-for="(k,i) in v.tag" :key="i">{{k}}</el-tag>
                    </div>
                </div>
            </div>
            <!-- 右侧 -->
            <div class="list-right">
                <div class="list-right-comment">
                    <div class="list-right-describe">
                        <div class="p1">好</div>
                
                        <a href="#">3333条点评</a>
                    </div>
                    <div class="list-right-score">
                        <p>4.5</p>
                    </div>
                </div>
                <div class="list-right-price">
                    <p>¥3333</p>
                     <el-button type="primary">查看详情</el-button>
                </div>
            </div>
        </li>
    </ul>
</div>
</template>

<script setup>
import {ref} from 'vue'

let recommendlist = ref([
    {id:1,name:'北京首都宾馆',img_url:'1.jpg',transport:'天安门/王府井 | 距市中心直线310米',starimg_url:'star5.png',tag:['网球场','商务中心','会议厅']},
    {id:2,name:'北京京伦饭店',img_url:'2.jpg',transport:'国贸CBD | 距市中心直线4.1公里',starimg_url:'star4.png',tag:['停车场','健身室','SPA']},
    {id:3,name:'北京新世界中心同派酒店',img_url:'3.jpg',transport:'前门/天坛公园/崇文门 | 距市中心直线1公里',starimg_url:'star4.png',tag:['桌球室','闪住','酒吧','行李寄存']},
    {id:4,name:'北京新世界中心同派酒店',img_url:'4.jpg',transport:'前门/天坛公园/崇文门 | 距市中心直线1公里',starimg_url:'star4.png',tag:['桌球室','闪住','酒吧','行李寄存']},
])
</script>

<style>

</style>

6. recommend.vue

<template>
  <div class="recommend">
    <div class="recommend-left">
        <h2>推荐酒店</h2>
        <Recommendlist></Recommendlist>
    </div>
    <div class="recommend-right">
        <Banner></Banner>
    </div>
  </div>
</template>

<script setup>
import { ref } from "vue";
import Banner from './banner.vue'
import Recommendlist from './recommend-list.vue'
</script>

<style>

</style>

7. toplist.vue

<template>
  <div class="topfilter">
    <div>
      <h4>位置区域</h4>
    </div>
    <!-- tabs切换 -->
    <div class="topfilter-tab">
      <el-tabs v-model="activeName" class="demo-tabs" @tab-click="handleClick">
        <!-- 数据绑定 -->
        <el-tab-pane :label="v.label" :name="v.name" v-for="v in topfilterlist" :key="v.id">
          <el-button  v-for="(k,i) in v.list" :key="i" size="small" type="info" plain>{{k}}</el-button>
        </el-tab-pane>
      </el-tabs>
    </div>
    <div>
      <h4>星级价格</h4>
    </div>
    <div class="topfilter-price">
        <el-button size="small" type="info" plain v-for="(v,i) in pricelist" :key="i">{{v}}</el-button>
    </div>
    <div>
      <h4>高级筛选</h4>
    </div>
    <div class="topfilter-filter">
        <el-dropdown v-for="(v,i) in filterlist" :key="i">
            <span class="el-dropdown-link">
              {{v.title}}
              <el-icon class="el-icon--right">
                <!-- 箭头按钮 -->
                <arrow-down />
              </el-icon>
            </span>
            <template #dropdown>
              <el-dropdown-menu>
                <el-dropdown-item v-for="(k,i) in v.list" :key="i">{{k}}</el-dropdown-item>
              </el-dropdown-menu>
            </template>
      </el-dropdown>
    </div>
  </div>
</template>

<script setup>
import {ref} from 'vue'
//箭头图标
import { ArrowDown } from '@element-plus/icons-vue'
const activeName = ref('first')
let topfilterlist = ref([
  {id:1,name:'first',label:'热门筛选',list:['天安门广场','王府井','北京大家','天安门广场','王府井','北京大家','天安门广场','王府井','北京大家','天安门广场','王府井','北京大家','天安门广场','王府井','北京大家','天安门广场','王府井','北京大家']},
  {id:2,name:'second',label:'机场车站',list:['北京南站','北京西站','北京火车站','北京南站','北京西站','北京火车站','北京南站','北京西站','北京火车站']},
  {id:3,name:'third',label:'商业区',list:['三里屯','国贸地区','西直门','三里屯','国贸地区','西直门','三里屯','国贸地区','西直门']},
  {id:4,name:'fourth',label:'景点',list:['什刹海','前门大街','五道口','什刹海','前门大街','五道口','什刹海','前门大街','五道口']}
])
let pricelist = ref(['五星(钻级)','四星(钻级)','三星(钻级)','两星(钻级)','一星(钻级)'])
let filterlist = ref([
  {title:'早餐',list:['含早餐','单份早餐','多份早餐']},
  {title:'支付方式',list:['在线付款','到店付款','闪住']},
  {title:'房型',list:['双床房','三床房','单人床房','大床房','特大床房']},
  {title:'酒店设施',list:['免费停车','室内游泳','单人床房','大床房','特大床房']},
  {title:'住宿类型',list:['免费停车','室内游泳','单人床房','大床房','特大床房']},
  {title:'酒特色主题',list:['免费停车','室内游泳','单人床房','大床房','特大床房']},
])
</script>

<style>

</style>

8. 最终效果

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/708124.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

经纬恒润助力微宏动力荣获ISO/SAE 21434网络安全流程认证证书

近日&#xff0c;经纬恒润与微宏动力合作的网络安全开发及认证项目顺利完成了阶段性里程碑。作为一家全球化的新能源及储电技术产品及解决方案供应商&#xff0c;微宏动力成功获得了由国际独立第三方检测、检验和认证机构UL Solutions授予的ISO/SAE 21434网络安全流程认证证书。…

社区团购系统搭建部署 :便捷高效,连接消费者与商家新篇章

一、前言 随着科技的快速发展和互联网的普及&#xff0c;社区团购系统作为一种新型的购物模式&#xff0c;正以其便捷高效的特性&#xff0c;逐渐改变着消费者和商家的互动方式。社区团购系统为商家提供丰富的营销活动和便捷高效的门店管理体系&#xff0c;为消费者提供真正实惠…

分享:2024年怎么做选品师项目才能赚钱?

在2024年&#xff0c;成为一名成功的选品师并赚钱&#xff0c;需要明确的策略和行动步骤。选品师作为电商行业中的关键角色&#xff0c;负责选择和管理产品库存&#xff0c;直接影响到销售和利润。以下是一些关键步骤&#xff0c;帮助您在这个领域取得成功。 1. 熟悉市场趋势和…

18、24年--信息系统工程——系统集成

1、集成基础 系统集成的工作再信息系统项目建设中非常重要,它通过硬件平台、网络通信平台、数据库平台、工具平台、应用软件平台将各类资源有机、高效地集成到一起,形成一个完整的工作台面。系统集成工作的好坏对系统开发、维护有极大的影响。因此,在技术上需要遵循的基本原…

labelme安装(通过anaconda)

1.下载安装anaconda 2.安装完成后打开&#xff0c;在环境页里面创建环境 选择3.6.13版本&#xff0c;然后运行 3.安装labelme pip install labelme -i https://pypi.tuna.tsinghua.edu.cn/simple 使用上面命令加速一下 4.labelme打开并标注 总结&#xff1a;现在版本越来越多…

WordPress如何删除内存中的缓存?

今天boke112百科将某篇文章修改分类和内容更新后&#xff0c;发现文章底部的相关文章显示的内容跟文章分类、标签毫无关系&#xff0c;还是显示原来的旧内容。后来查看YIA主题相关文章的代码&#xff0c;才发现相关文章的数据保存到内存中的&#xff0c;而且是永不过期&#xf…

SAP BW:传输转换源系统-源系统映射关系

最近有朋友再问问我源系统映射关系怎么配置&#xff0c;想着写一个怕以后忘了。 简单说下这个是干嘛的&#xff0c;其实就是配置一个源系统到目标系统的一个映射&#xff0c;这样传输的时候才知道传过来的数据源要变成目标系统的数据源。 比如下图&#xff0c;在开发环境&…

视频去水印用哪个软件

在数字化时代的浪潮中&#xff0c;视频内容已成为我们生活中不可或缺的一部分。然而&#xff0c;随着视频的广泛传播&#xff0c;水印问题也逐渐浮出水面。水印影响了视频的美观度&#xff0c;本文将为您详细介绍视频去水印的常用方法&#xff0c;帮助您轻松解决水印问题。 搜索…

pytest中失败用例重跑

pip install pytest-rerunfailures 下载rerunfailures插件包 配置文件中加入命令 --reruns 次数 也可在命令行中pytest --rerun-failures2 可以在allure报告中看到重试效果

利用穿戴甲虚拟试戴技术提高销量和参与度

在不断变化的美容行业&#xff0c;保持领先意味着拥抱创新技术。其中一项改变游戏规则的技术是人工智能驱动的虚拟指甲试戴。在穿戴甲领域&#xff0c;不断兴起的虚拟试戴技术对促进销售和参与度产生了重大影响。 视觉吸引力的力量 要了解虚拟试戴的重要性&#xff0c;必须了解…

【2024重启篇】ESP32的乱搞系列

按视频教程看自己能不能做成功&#xff08;问题多多博客&#xff09; 现在时间2024.6.13日 手头上有2022年到手的&#xff1a; ESP32-S&#xff08;双核&#xff09;的ESP32-CAM摄像头SD卡板1块&#xff08;还有一块收藏了&#xff09;。 ESP8266&#xff08;是单核&#xff0…

Axure中继器交互效果

部件库预览链接&#xff1a; https://ezd11a.axshare.com&#xff08;请与班主任联系获取原型文档&#xff09; 支持版本: Axrure RP 8 文件大小: 109KB 文档内容介绍 “翻页”效果 “排序”效果 “全反选”效果 “筛选”效果 “删除”效果 免费领取资料 添加班主任回复 “…

Guitar Pro 8中文版安装包下载及安装教程

Guitar Pro是一款倍受吉他手喜爱的吉他和弦、六线谱、BASS四线谱绘制、打印、查看、试听软件&#xff0c;它也是一款优秀的MIDI音序器&#xff0c;MIDI制作辅助工具&#xff0c;可以输出标准格式的MIDI。 GP的过人之处就在于它可以直接用鼠标和键盘按标准的六线谱、四线谱进行…

OpenCV读取和显示和保存图像

# 导入 OpenCV import cv2 as cv # 读取图像 image cv.imread(F:\\mytupian\\xihuduanqiao.jpg) # 创建窗口 #显示图像后&#xff0c;允许用户随意调整窗口大小 cv.namedWindow(image, cv.WINDOW_NORMAL) # 显示图像 cv.imshow(image, image)# 将图像保存到文件 success cv…

vue2 + element-ui,前端配置化表单封装(2024-06-14)

技术栈是 vue2 element-ui&#xff0c;主要能解决的问题就是 提高代码复用能力、提升开发效率&#xff0c;特别是需要开发多个大型表单系统的&#xff0c;配置化可以极大的提升效率&#xff0c;让你上班摸鱼不再是梦想&#xff01;为了早点下班&#xff0c;我们接着往下看吧&a…

舵机是什么?舵机内部结构解析

什么是舵机呢&#xff1f;首先&#xff0c;“舵机”这个名号其实是一个俗称&#xff0c;是那些玩航模、船模的人起的名字&#xff0c;因为这种电机常被用于舵面控制。舵机其实就是一个简单的伺服电机系统&#xff0c;也是最常见的伺服电机系统。 舵机是一种位置&#xff08;角…

领夹无线麦克风哪个品牌好?分享麦克风什么牌子的音质比较好!

​无线领夹麦克风作为现代音频技术的杰出代表&#xff0c;正逐渐改变着我们的声音表达方式。它以其独特的便携性、稳定性和高音质&#xff0c;成为了众多声音创作者的首选工具。面对市场上琳琅满目的无线领夹麦克风选项&#xff0c;找到一款贴合个人需求的设备无疑是一项挑战。…

在开发盲盒App小程序时,需要注意哪些地方!

在开发盲盒App小程序时&#xff0c;需要注意以下几个方面&#xff0c;以确保项目的顺利进行和最终产品的成功&#xff1a; 一、明确开发目标和功能 确定小程序的定位&#xff0c;明确主要面向的消费群体和市场需求。根据目标用户群体&#xff0c;列出小程序所需的主要功能&am…

1_常见指令【Linux中常见指令的学习和使用】【万字长文】

常见指令以及权限理解 开始学习linux前的注意事项 在学习linux之前&#xff0c;我们要知道linux是一个操作系统。 那操作系统是什么呢&#xff1f;&#xff08;这里只做大概了解&#xff09; 操作系统就是一个管理软硬件的软件。 它对上提供良好&#xff08;稳定、高效、安…

重装了mysql,然后安装为服务时,net start 启动一直报错,MySQL服务无法启动的解决

之前写过一篇关于安装mysql的文章&#xff0c;按上面的处理&#xff0c;基本上是可以的。 今天换了下目录&#xff0c;重新安装&#xff0c;一直报错。 然后我们来看一下问题&#xff1a; mysqld -console 这里的目录是有问题的&#xff0c;设置的是&#xff1a;datadird:\to…