系列文章目录

Vue基础篇一:编写第一个Vue程序
Vue基础篇二:Vue组件的核心概念
Vue基础篇三:Vue的计算属性与侦听器
Vue基础篇四:Vue的生命周期(秒杀案例实战)
Vue基础篇五:Vue的指令
Vue基础篇六:Vue使用JSX进行动态渲染
Vue提高篇一:使用Vuex进行状态管理
Vue提高篇二:使用vue-router实现静态路由
Vue提高篇三:使用vue-router实现动态路由
Vue提高篇四:使用Element UI组件库
Vue提高篇五:使用Jest进行单元测试
Vue提高篇六: 使用Vetur+ESLint+Prettier插件提升开发效率
Vue实战篇一: 使用Vue搭建注册登录界面
Vue实战篇二: 实现邮件验证码发送
Vue实战篇三:实现用户注册
Vue实战篇四:创建多步骤表单
Vue实战篇五:实现文件上传
Vue实战篇六:表格渲染动态数据
Vue实战篇七:表单校验
Vue实战篇八:实现弹出对话框进行交互
Vue实战篇九:使用省市区级联选择插件
Vue实战篇十:响应式布局
Vue实战篇十一:父组件获取子组件数据的常规方法
Vue实战篇十二:多项选择器的实际运用
Vue实战篇十三:实战分页组件
Vue实战篇十四:前端excel组件实现数据导入
Vue实战篇十五:表格数据多选在实际项目中的技巧
Vue实战篇十六:导航菜单
Vue实战篇十七:用树型组件实现一个知识目录
Vue实战篇十八:搭建一个知识库框架
Vue实战篇十九:使用printjs打印表单
Vue实战篇二十:自定义表格合计
Vue实战篇二十一:实战Prop的双向绑定
Vue实战篇二十二:生成二维码
Vue实战篇二十三:卡片风格与列表风格的切换
Vue实战篇二十四:分页显示
Vue实战篇二十五:使用ECharts绘制疫情折线图
Vue实战篇二十六:创建动态仪表盘
Vue实战篇二十七:实现走马灯效果的商品轮播图
Vue实战篇二十八:实现一个手机版的购物车
Vue实战篇二十九:模拟一个简易留言板
Vue项目实战篇一:实现一个完整的留言板(带前后端源码下载)
Vue实战篇三十:实现一个简易版的头条新闻

文章目录

  • 系列文章目录
  • 一、背景
  • 二、制作频道组件
    • 2.1 获取新闻频道数据
    • 2.2 组件设计
  • 三、改造主页
  • 四、效果演示

一、背景

  • 在上一篇文章中,我们实现了一个简易版的头条新闻,这次我们做个改进。
  • 加入新闻频道,可以让用户选择不同的频道,阅读新闻。

二、制作频道组件

2.1 获取新闻频道数据

  • 我们仍然通过极数数据接口,来获取新闻频道的数据
  • 编写前端Api
import axios from 'axios'axios.defaults.baseURL = '/apis'// 向极速数据免费新闻接口获取新闻频道
export function getNewChannel() {return new Promise((resolve, reject) => {axios.get('/news/channel?appkey=自己在极速数据上申请的appkey').then(res => {resolve(res)}).catch(error => { reject(error) })})
}

2.2 组件设计

  • 我们需要单独编写一个频道组件,主要包含以下这些功能:
    1、横向展示频道列表

    2、可左右滑动

    3、选择频道后,获取频道对应的新闻列表

    4、将选择的频道、新闻列表、刷新状态放入状态管理器中存储,供主页刷新调用
  • 以下是频道组件的完整代码
<template><div class="channel-box"><div class="channel-list"><ul ref="channelList" :style="{ left: -nowLeft + 'px' }"><li v-for="(item, index) in list" :key="index" :ref="'li' + index"><span class="channel" :class="{ 'channel-active': id === item.id }" @click="selectChannel(item)">{{ item.tag_name }}</span></li></ul></div><div class="icon-lf"><i v-show="showLf" class="el-icon-arrow-left" @click="handleLeft" /></div><div class="icon-rt"><i v-show="showRt" class="el-icon-arrow-right" @click="handleRight" /></div></div>
</template><script>
import { getNewChannel, getNewList } from '@/api/news'
export default {data() {return {list: [],id: 0,fixedWidth: 200,nowNum: 0,showLf: false,showRt: false,allWidth: 0,nowLeft: 0,nowIndex: 0 }},mounted() {this.getChannel().then(res => {console.log('channel', res)if (res && res.status === 200 && res.data.status === 0) {this.list = []res.data.result.forEach((element, index) => {this.list.push({ 'id': index, 'tag_name': element })})this.$nextTick(() => {this.allWidth = this.$refs.channelList.offsetWidthif (this.allWidth > this.fixedWidth) {this.showRt = true}this.selectChannel({ id: 0, tag_name: '头条' })})}})},methods: {// 异步获取频道async getChannel() {const data = await getNewChannel()return data},// 选择频道,根据频道获取新闻selectChannel(item) {this.$store.commit('SET_LOADING', true)this.$store.commit('SET_CHANNEL', item.tag_name)this.getNews(item.tag_name).then(res => {console.log('news', res)if (res) {scrollTo(0, 0)this.$store.commit('SET_NEWS', res.data.result.list)this.id = item.id}this.$store.commit('SET_LOADING', false)})},// 异步获取新闻async getNews(channel) {const data = await getNewList(channel)return data},// 频道列表左移handleLeft() {if (this.nowLeft > 0) {this.nowNum--this.showRt = trueif (this.nowNum > 0) {let nw = 0for (let j = this.list.length; j >= 0; j--) {if (j < this.nowIndex) {nw += this.$refs['li' + j][0].offsetWidthif (nw >= this.fixedWidth) {nw -= this.$refs['li' + j][0].offsetWidththis.nowLeft -= nwthis.nowIndex = j + 1break}}}} else {this.nowLeft = 0this.nowIndex = 0this.showLf = false}}},// 频道列表右移handleRight() {if (this.nowLeft + this.fixedWidth < this.allWidth) {this.nowNum++this.showLf = truelet nw = 0for (let i = 0; i < this.list.length; i++) {if (i >= this.nowIndex) {nw += this.$refs['li' + i][0].offsetWidthif (nw > this.fixedWidth) {nw -= this.$refs['li' + i][0].offsetWidththis.nowLeft += nwthis.nowIndex = ibreak}}}if (this.nowLeft + this.fixedWidth >= this.allWidth) {this.showRt = false}}}}
}
</script><style lang="scss" scoped>
.channel-box {width: 100%;padding: 0 20px;height: 46px;position: fixed;align-items: center;top: 1.2rem;font-size: 18px;letter-spacing: 3px;background-color: rgb(252, 248, 248);.channel-list {width: 100%;height: 100%;overflow: hidden;position: relative;margin-top: 0.2rem;ul {transition-duration: 0.3s;position: absolute;top: 0px;left: 0px;margin: 0;padding: 0;display: flex;flex-wrap: nowrap;li {white-space: nowrap;display: inline-block;white-space: nowrap;padding: 0 10px;}li:first-child {padding-left: 0;}li:last-child {padding-right: 0;}}.channel {cursor: pointer;display: inline-block;height: 28px;line-height: 28px;transition: border-color 0.2s;&:hover {color: #e72521;}}.channel-active {color: #e72521;}}.icon-lf {cursor: pointer;line-height: 30px;position: absolute;left: 5px;top: 6px;}.icon-rt {line-height: 30px;cursor: pointer;position: absolute;right: 5px;top: 6px;}
}
</style>

三、改造主页

  • 我们将频道组件,放入主页,进行改造
    1、主页判断状态管理器的刷新状态,如果刷新状态为true,则显示加载页面

2、主页判断状态管理器的刷新状态,如果刷新状态为false,则显示新闻列表

  • 以下是改造后的完整源码:
<template><div><!-- 标题栏 --><div class="header"><span /><span>头条新闻</span><span /></div><channel /><!-- 新闻列表 --><div class="nav-content"><div v-if="loading == false" class="newsContent"><divv-for="(item, index) in newData":key="index"class="section"@click="toNews(index)"><div class="news"><div class="news-left"><img :src="item.pic" alt=""></div><div class="news-right"><div class="newsTitle">{{ item.title }}</div><div class="newsMessage"><span>{{ item.time }}</span><span>{{ item.src }}</span></div></div></div></div></div><el-mainv-elsev-loading="loading"class="load"element-loading-background="rgba(0,0,0,0)"element-loading-text="正在加载中"/></div></div>
</template><script>
import Channel from './channel'
export default {name: 'Home',components: { Channel },data() {return {}},computed: {newData() {return this.$store.state.news.newsData},loading() {return this.$store.state.news.loading}},methods: {toNews(index) {this.$store.commit('SET_NEWS_INDEX', index)this.$router.push('/news')}}}
</script><style lang="scss"  scoped>
.header {width: 100%;height: 1.2rem;background-color: #d43d3d;display: flex;justify-content: space-between;align-items: center;color: #fff;font-size: 20px;font-weight: 700;letter-spacing: 3px;z-index: 99;position: fixed;top: 0;img {width: 0.67rem;height: 0.67rem;cursor: pointer;}
}.nav-content {margin-top: 2.5rem;
}
.nav {width: 100%;height: 0.96rem;background-color: #f4f5f6;display: flex;position: fixed;z-index: 98;
}.section {width: 100%;height: 2.5rem;border-bottom: 1px solid #ccc;
}
.newsContent {padding-top: 0rem;
}
.news {height: 2.25rem;box-sizing: border-box;margin: 10px 10px;display: flex;
}
.news-left {height: 100%;width: 2.8rem;display: inline-block;
}
.news-left img {width: 100%;height: 100%;
}
.news-right {flex: 1;padding-left: 10px;
}
.newsTitle {width: 100%;height: 62%;color: #404040;font-size: 17px;overflow: hidden;
}
.newsMessage {width: 100%;height: 38%;display: flex;align-items: flex-end;color: #888;justify-content: space-between;
}
.load {width: 100%;height: 100%;overflow: hidden;
}
</style>

四、效果演示

Vue实战篇三十一:实现一个改进版的头条新闻相关推荐

  1. Vue实战篇三十三:实现新闻的浏览历史

    系列文章目录 Vue基础篇一:编写第一个Vue程序 Vue基础篇二:Vue组件的核心概念 Vue基础篇三:Vue的计算属性与侦听器 Vue基础篇四:Vue的生命周期(秒杀案例实战) Vue基础篇五:V ...

  2. Vue实战篇三十:实现一个简易版的头条新闻

    系列文章目录 Vue基础篇一:编写第一个Vue程序 Vue基础篇二:Vue组件的核心概念 Vue基础篇三:Vue的计算属性与侦听器 Vue基础篇四:Vue的生命周期(秒杀案例实战) Vue基础篇五:V ...

  3. Vue实战篇二十八:实现一个手机版的购物车

    系列文章目录 Vue基础篇一:编写第一个Vue程序 Vue基础篇二:Vue组件的核心概念 Vue基础篇三:Vue的计算属性与侦听器 Vue基础篇四:Vue的生命周期(秒杀案例实战) Vue基础篇五:V ...

  4. Vue实战篇二十九:模拟一个简易留言板

    系列文章目录 Vue基础篇一:编写第一个Vue程序 Vue基础篇二:Vue组件的核心概念 Vue基础篇三:Vue的计算属性与侦听器 Vue基础篇四:Vue的生命周期(秒杀案例实战) Vue基础篇五:V ...

  5. Vue实战篇十七:用树型组件实现一个文档目录结构

    系列文章目录 Vue基础篇一:编写第一个Vue程序 Vue基础篇二:Vue组件的核心概念 Vue基础篇三:Vue的计算属性与侦听器 Vue基础篇四:Vue的生命周期(秒杀案例实战) Vue基础篇五:V ...

  6. Vue实战篇二十七:实现走马灯效果的商品轮播图

    系列文章目录 Vue基础篇一:编写第一个Vue程序 Vue基础篇二:Vue组件的核心概念 Vue基础篇三:Vue的计算属性与侦听器 Vue基础篇四:Vue的生命周期(秒杀案例实战) Vue基础篇五:V ...

  7. Vue实战篇二十六:创建动态仪表盘

    系列文章目录 Vue基础篇一:编写第一个Vue程序 Vue基础篇二:Vue组件的核心概念 Vue基础篇三:Vue的计算属性与侦听器 Vue基础篇四:Vue的生命周期(秒杀案例实战) Vue基础篇五:V ...

  8. Vue实战篇十六:导航菜单

    系列文章目录 Vue基础篇一:编写第一个Vue程序 Vue基础篇二:Vue组件的核心概念 Vue基础篇三:Vue的计算属性与侦听器 Vue基础篇四:Vue的生命周期(秒杀案例实战) Vue基础篇五:V ...

  9. Vue实战篇五:实现文件上传

    系列文章目录 Vue基础篇一:编写第一个Vue程序 Vue基础篇二:Vue组件的核心概念 Vue基础篇三:Vue的计算属性与侦听器 Vue基础篇四:Vue的生命周期(秒杀案例实战) Vue基础篇五:V ...

最新文章

  1. 近世代数--置换群--置换permutation分解成什么?置换的级如何计算?
  2. mysql中的字典项是啥_常见数据库设计(1)——字典数据
  3. Session.Abandon和Session.Clear有何不同
  4. pyperclip模块
  5. python第一周心得体会_Python学习第一周学习总结
  6. mysql设置不区分大小写
  7. C语言--第2次作业
  8. informix的常用SQL语句
  9. C# 程序异常关闭时的捕获
  10. 字符串重复字符统计c语言,[讨论]精简字符串重复字符程序
  11. matlab 计算图像的基本参数
  12. HIT计算机系统大作业——hello的一生
  13. php 写聊天室源码,php写的简易聊天室代码
  14. html设置flash的背景颜色,flash 如何设置舞台大小和背景颜色?
  15. vyos配置src-nat
  16. python中怎么查看二进制转换的数据_python二进制数据
  17. python数据分析(六)—数据清洗2
  18. sublime text 光标移动行末/行首
  19. 群晖|半洗白后moments正常显示人像、主题、预览
  20. 生活实用之清洗小龙虾

热门文章

  1. 淘宝客优惠券领取微信小程序前台和后台C#源码
  2. dumps-loads dump-load的区别
  3. 游戏音乐制作的硬件环境
  4. win10如何使用miphone刷机工具
  5. 美菜Java笔试,美菜Java面经(一面+二面)
  6. 基于微信平台做视频直播时,如何裂变引流吸粉
  7. 时尚渐变职业生涯规划PPT模板
  8. dota 火猫TC二连技巧
  9. 完美,腾讯技术官发布Java零基础就业宝典,不用再怀疑人生了
  10. veryCD关闭之后,我们该何去何从。别急,找这里。