【vue】三、vue2仿去哪儿app——城市列表页面

2024-01-14 01:59

本文主要是介绍【vue】三、vue2仿去哪儿app——城市列表页面,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • 三、vue2仿去哪儿app——城市列表页面
      • Ⅰ 页面结构
      • Ⅱ 开发笔记及注意点
          • 1.使用Better-scroll第三方包实现拖动
          • 2.1. 使用router-link实现页面的跳转
          • 2.2 router点击后变色问题
          • 3.实现点击字母表某个字母,跳转到相应字母的城市列表项(兄弟组件间联动)
          • 4.在左侧字母表中做上下拖拽的时候,城市列表内容也跟着变化
          • 5.Vuex的高级使用及localStorage
          • 6.使用函数节流,限制move方法的频率
          • 7.使用keep-alive优化网页性能

三、vue2仿去哪儿app——城市列表页面

Ⅰ 页面结构

城市列表页面

  • 搜索框
  • 热门城市
  • 城市列表
  • 右侧字母表导航

在这里插入图片描述

Ⅱ 开发笔记及注意点

1.使用Better-scroll第三方包实现拖动
npm install better-scroll --save

引入better-scroll:import Bscroll from 'better-scroll'

使用vue中的ref属性来获取DOM元素。
在List.vue中,总组件需要有一个div包裹住其他的组件,并且在这个div上加上ref属性:

<div class="list" ref="wrapper">

使用mounted()函数,在页面DOM挂载完毕的时候,创建better-scroll实例属性,并将wrapper传进去:

mounted () {this.scroll = new Bscroll(this.$refs.wrapper, {click: true})}

注意:在使用better-scroll时,会使click事件失效
原因:better-scroll 默认会阻止浏览器的原生 click 事件。
因此,需要我们在创建实例属性时,将click属性设置为true。

2.1. 使用router-link实现页面的跳转
<router-link to='/city'>
//点击后跳转的控件
</router-link>

注意:加了这个组件之后,需要重新设置字体颜色,因为组件中包含一个链接,链接本身设置了字体颜色。

2.2 router点击后变色问题

因为router-link也有超链接样式,点击之后会变色,所以要给这一层的盒子加一个固定的颜色color。否则点击过后颜色就改变了

3.实现点击字母表某个字母,跳转到相应字母的城市列表项(兄弟组件间联动)

首先,要实现兄弟组件间的传值,则可以采取以下方法:
(1)首先,将Alphabet.vue的letter值传递给父组件City.vue;
(2)接着,父组件接收子组件Alphabet.vue传递过来的值,并将该值传递给子组件List.vue

在Alphabet.vue中,绑定click事件函数,在事件函数中,使用$e在这里插入代码片mit()函数触发父组件自定义事件,并将letter值传过去:
(1)div上:@click="handleLetterClick"
(2)事件函数:

handleLetterClick (e) {this.$emit('change', e.target.innerText)},

父组件City.vue中,在data中定义letter属性,并设值为空,在change事件所绑定的函数中,接收letter值,并设置为自定义的letter值,最后将自定义的letter值传递给子组件List.vue:
(1)change事件绑定函数:

<city-alphabet :cities='cities' @change="handleLetterChange"></city-alphabet>

(2)data中定义letter值:

data () {return {letter: ''}},

(3)事件函数:

handleLetterChange (letter) {this.letter = letter}

(4)将自定义的letter值传递给子组件List.vue:

<city-list :hot='hotCities' :cities='cities' :letter="letter"></city-list>

子组件List.vue中,接收父组件传来的letter值,借助监听器,当letter值改变时,调用better-scroll提供的接口:scrollToElement():让better-scroll自动滚到某个元素上。
(1)在props接收父组件的值:

props: {letter: String},

(2)借助监听器,监听letter值的改变:只要letter发生变化,就会执行该函数。
首先,在列表区域通过ref属性获取相应字母列表DOM元素:

<div class="area" v-for="(item,key) of cities" :key="key" :ref="key">

接着,在watch监听器中,获取当前字母的城市列表,

调用better-scroll提供的接口:scrollToElement():让better-scroll自动滚到字母对应的城市列表上:

watch: {letter () {if (this.letter) {const element = this.$refs[this.letter][0]this.scroll.scrollToElement(element)}}},
4.在左侧字母表中做上下拖拽的时候,城市列表内容也跟着变化

主要是监听拖拽时间,在li增加了三个事件,分别是touchstart、touchmove、touchend来监听拖拽事件,同时在li上使用ref,在data定义了一个touchStatus来保存是否在拖拽的状态,并使用计算属性把cities中的字母取出来放在数组letters中,最后计算拖拽到哪个字母:

    handleTouchStart () {this.touchStatus = true},handleTouchMove (e) {if(this.touchStatus){const startY = this.$refs['A'][0].offsetTopconst touchY = e.touches[0].clientY -79const index = Math.floor((touchY-startY) / 20)if(index >= 0 && index < this.letters.length) {this.$emit('change',this.letters[index])}}},handleTouchEnd () {this.touchStatus = false}

Alphabet.vue:

<template><ul class="list"><li class="item" v-for="item of letters" :key="item":ref="item"@touchstart="handleTouchStart"@touchmove="handleTouchMove"@touchend="handleTouchEnd"@click="handleLetterClick">{{item}}</li></ul>
</template>
<script>
export default {name:'CityAlphabet',props: {cities: Object},data () {return {touchStatus: false}},computed: {letters () {const letters = []for( let i in this.cities) {letters.push(i)}return letters}},methods: {handleLetterClick (e) {this.$emit('change',e.target.innerText)},handleTouchStart () {this.touchStatus = true},handleTouchMove (e) {if(this.touchStatus){const startY = this.$refs['A'][0].offsetTopconst touchY = e.touches[0].clientY -79const index = Math.floor((touchY-startY) / 20)if(index >= 0 && index < this.letters.length) {this.$emit('change',this.letters[index])}}},handleTouchEnd () {this.touchStatus = false}}
}
</script>
<style lang="stylus" scoped>@import '~styles/varibles.styl'.listdisplay flexflex-direction columnjustify-content centerposition absolutetop 1.58remright 0bottom 0width .4rem.itemline-height .4remtext-align centercolor $bgColor
</style>
5.Vuex的高级使用及localStorage
import Vue from 'vue'
import Vuex from 'vuex'Vue.use(Vuex)export default new Vuex.Store({state: {city: localStorage.city || '成都'},mutations: {changeCity (state, city) {state.city = citylocalStorage.city = city}}
})

上面实现的功能会有问题,就是每次重启应用的时候,选择的城市会重置为State中预设的城市名,为了解决这个问题,在这里就引入了h5中的localStorage本地缓存来实现

import Vue from 'vue'
import Vuex from 'vuex'Vue.use(Vuex)let defaultCity = '上海'
try {if (localStorage.city) {defaultCity = localStorage.city}
} catch (e) {}export default new Vuex.Store({state: {city: defaultCity},mutations: {changeCity (state, city) {state.city = citytry {localStorage.city = city} catch (e) {}}}
})
6.使用函数节流,限制move方法的频率
  1. 首先,在data中定义timer=null
  2. 在handleTouchMove ()方法中,当timer不为空时,清除timer;若为空,则定义定时器:
handleTouchMove (e) {if (this.touchStatus) {if (this.timer) {clearTimeout(this.timer)}this.timer = setTimeout(() => {const touchY = e.touches[0].clientY - 79const index = Math.floor((touchY - this.startY) / 20)if (index >= 0 && index < this.letters.length) {this.$emit('change', this.letters[index])}}, 10)}},
7.使用keep-alive优化网页性能

当前页面存在一个问题:每次路由切换,总会发送一次ajax请求(每次切换,页面总会重新渲染,mounted钩子重新执行,ajax数据就会重新获取)。
为了优化网页的性能,使用keep-alive:实现路由加载一次过后,会将加载内容放入内存之中。下一次进去,不需要重新渲染这个组件,mounted()函数也不再执行(即不再发送ajax请求),只需要从内存将内容取出显示

<keep-alive exclude="Detail"><router-view/>
</keep-alive>

在使用keep-alive之后,重新刷新页面时,mounted()函数都不会再次执行,导致如果改变城市后,首页其他数据没有相应的重新获取以及改变。

还有逻辑问题,就是在重新选择城市后在Home.vue中需要重新进行Ajax请求,

在Home.vue中引入vuex

import { mapState } from 'vuex'

修改getHomeinfo方法,在请求的时候带上state中的city

getHomeinfo () {axios.get('/api/index.json?city=' + this.city).then(this.getHomeinfoSucc)
},
activated () {if (this.lastCity !== this.city) {this.lastCity = this.citythis.getHomeInfo()}}

这篇关于【vue】三、vue2仿去哪儿app——城市列表页面的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/603572

相关文章

HTML5中的Microdata与历史记录管理详解

《HTML5中的Microdata与历史记录管理详解》Microdata作为HTML5新增的一个特性,它允许开发者在HTML文档中添加更多的语义信息,以便于搜索引擎和浏览器更好地理解页面内容,本文将探... 目录html5中的Mijscrodata与历史记录管理背景简介html5中的Microdata使用M

html5的响应式布局的方法示例详解

《html5的响应式布局的方法示例详解》:本文主要介绍了HTML5中使用媒体查询和Flexbox进行响应式布局的方法,简要介绍了CSSGrid布局的基础知识和如何实现自动换行的网格布局,详细内容请阅读本文,希望能对你有所帮助... 一 使用媒体查询响应式布局        使用的参数@media这是常用的

HTML5表格语法格式详解

《HTML5表格语法格式详解》在HTML语法中,表格主要通过table、tr和td3个标签构成,本文通过实例代码讲解HTML5表格语法格式,感兴趣的朋友一起看看吧... 目录一、表格1.表格语法格式2.表格属性 3.例子二、不规则表格1.跨行2.跨列3.例子一、表格在html语法中,表格主要通过< tab

Vue3组件中getCurrentInstance()获取App实例,但是返回null的解决方案

《Vue3组件中getCurrentInstance()获取App实例,但是返回null的解决方案》:本文主要介绍Vue3组件中getCurrentInstance()获取App实例,但是返回nu... 目录vue3组件中getCurrentInstajavascriptnce()获取App实例,但是返回n

Python列表去重的4种核心方法与实战指南详解

《Python列表去重的4种核心方法与实战指南详解》在Python开发中,处理列表数据时经常需要去除重复元素,本文将详细介绍4种最实用的列表去重方法,有需要的小伙伴可以根据自己的需要进行选择... 目录方法1:集合(set)去重法(最快速)方法2:顺序遍历法(保持顺序)方法3:副本删除法(原地修改)方法4:

如何解决idea的Module:‘:app‘platform‘android-32‘not found.问题

《如何解决idea的Module:‘:app‘platform‘android-32‘notfound.问题》:本文主要介绍如何解决idea的Module:‘:app‘platform‘andr... 目录idea的Module:‘:app‘pwww.chinasem.cnlatform‘android-32

JS+HTML实现在线图片水印添加工具

《JS+HTML实现在线图片水印添加工具》在社交媒体和内容创作日益频繁的今天,如何保护原创内容、展示品牌身份成了一个不得不面对的问题,本文将实现一个完全基于HTML+CSS构建的现代化图片水印在线工具... 目录概述功能亮点使用方法技术解析延伸思考运行效果项目源码下载总结概述在社交媒体和内容创作日益频繁的

前端CSS Grid 布局示例详解

《前端CSSGrid布局示例详解》CSSGrid是一种二维布局系统,可以同时控制行和列,相比Flex(一维布局),更适合用在整体页面布局或复杂模块结构中,:本文主要介绍前端CSSGri... 目录css Grid 布局详解(通俗易懂版)一、概述二、基础概念三、创建 Grid 容器四、定义网格行和列五、设置行

前端下载文件时如何后端返回的文件流一些常见方法

《前端下载文件时如何后端返回的文件流一些常见方法》:本文主要介绍前端下载文件时如何后端返回的文件流一些常见方法,包括使用Blob和URL.createObjectURL创建下载链接,以及处理带有C... 目录1. 使用 Blob 和 URL.createObjectURL 创建下载链接例子:使用 Blob

Vuex Actions多参数传递的解决方案

《VuexActions多参数传递的解决方案》在Vuex中,actions的设计默认只支持单个参数传递,这有时会限制我们的使用场景,下面我将详细介绍几种处理多参数传递的解决方案,从基础到高级,... 目录一、对象封装法(推荐)二、参数解构法三、柯里化函数法四、Payload 工厂函数五、TypeScript