【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

相关文章

spring-boot-starter-thymeleaf加载外部html文件方式

《spring-boot-starter-thymeleaf加载外部html文件方式》本文介绍了在SpringMVC中使用Thymeleaf模板引擎加载外部HTML文件的方法,以及在SpringBoo... 目录1.Thymeleaf介绍2.springboot使用thymeleaf2.1.引入spring

部署Vue项目到服务器后404错误的原因及解决方案

《部署Vue项目到服务器后404错误的原因及解决方案》文章介绍了Vue项目部署步骤以及404错误的解决方案,部署步骤包括构建项目、上传文件、配置Web服务器、重启Nginx和访问域名,404错误通常是... 目录一、vue项目部署步骤二、404错误原因及解决方案错误场景原因分析解决方案一、Vue项目部署步骤

前端原生js实现拖拽排课效果实例

《前端原生js实现拖拽排课效果实例》:本文主要介绍如何实现一个简单的课程表拖拽功能,通过HTML、CSS和JavaScript的配合,我们实现了课程项的拖拽、放置和显示功能,文中通过实例代码介绍的... 目录1. 效果展示2. 效果分析2.1 关键点2.2 实现方法3. 代码实现3.1 html部分3.2

Python如何计算两个不同类型列表的相似度

《Python如何计算两个不同类型列表的相似度》在编程中,经常需要比较两个列表的相似度,尤其是当这两个列表包含不同类型的元素时,下面小编就来讲讲如何使用Python计算两个不同类型列表的相似度吧... 目录摘要引言数字类型相似度欧几里得距离曼哈顿距离字符串类型相似度Levenshtein距离Jaccard相

CSS弹性布局常用设置方式

《CSS弹性布局常用设置方式》文章总结了CSS布局与样式的常用属性和技巧,包括视口单位、弹性盒子布局、浮动元素、背景和边框样式、文本和阴影效果、溢出隐藏、定位以及背景渐变等,通过这些技巧,可以实现复杂... 一、单位元素vm 1vm 为视口的1%vh 视口高的1%vmin 参照长边vmax 参照长边re

CSS3中使用flex和grid实现等高元素布局的示例代码

《CSS3中使用flex和grid实现等高元素布局的示例代码》:本文主要介绍了使用CSS3中的Flexbox和Grid布局实现等高元素布局的方法,通过简单的两列实现、每行放置3列以及全部代码的展示,展示了这两种布局方式的实现细节和效果,详细内容请阅读本文,希望能对你有所帮助... 过往的实现方法是使用浮动加

css渐变色背景|<gradient示例详解

《css渐变色背景|<gradient示例详解》CSS渐变是一种从一种颜色平滑过渡到另一种颜色的效果,可以作为元素的背景,它包括线性渐变、径向渐变和锥形渐变,本文介绍css渐变色背景|<gradien... 使用渐变色作为背景可以直接将渐China编程变色用作元素的背景,可以看做是一种特殊的背景图片。(是作为背

Redis存储的列表分页和检索的实现方法

《Redis存储的列表分页和检索的实现方法》在Redis中,列表(List)是一种有序的数据结构,通常用于存储一系列元素,由于列表是有序的,可以通过索引来访问元素,因此可以很方便地实现分页和检索功能,... 目录一、Redis 列表的基本操作二、分页实现三、检索实现3.1 方法 1:客户端过滤3.2 方法

CSS自定义浏览器滚动条样式完整代码

《CSS自定义浏览器滚动条样式完整代码》:本文主要介绍了如何使用CSS自定义浏览器滚动条的样式,包括隐藏滚动条的角落、设置滚动条的基本样式、轨道样式和滑块样式,并提供了完整的CSS代码示例,通过这些技巧,你可以为你的网站添加个性化的滚动条样式,从而提升用户体验,详细内容请阅读本文,希望能对你有所帮助...

css实现图片旋转功能

《css实现图片旋转功能》:本文主要介绍了四种CSS变换效果:图片旋转90度、水平翻转、垂直翻转,并附带了相应的代码示例,详细内容请阅读本文,希望能对你有所帮助... 一 css实现图片旋转90度.icon{ -moz-transform:rotate(-90deg); -webkit-transfo