uniapp实战仿写网易云音乐(二)—promise接口请求的封装和主页功能的实现,组件封装

news2025/1/6 19:05:46

文章目录

    • 前言
    • promise请求接口的封装
    • 主入口功能的实现:
    • 推荐歌单模块
    • 新碟新歌模块
    • 精选视频模块
    • 最后

前言

本篇文章继续完成上篇文章的部分,主要实现prromise接口的封装和首页主入口的实现

promise请求接口的封装

在上篇文章中请求我们是这样写的:

methods: {
	getBanner() {
		uni.request({
			url: 'http://localhost:3000/banner',
			method: 'GET',
			success:(res)=>{
				this.swiper = res.data.banners;
			}
		})
	}
}

但实际开发中,接口非常多,这样又不方便管理,所以我们通常会进行接口的封装
先新建util文件夹中request.js文件,在文件中进行promise的封装:
在这里插入图片描述

封装里面的状态码根据实际情况写,我这里写简单写了几个

/**
 * 请求组件封装
 * @param {Object} url 请求地址 /banner
 * @param {Object} data 请求参数
 * @param {Object} method 请求的方法
 * @param {Object} contentType 请求内容类型 1=json  2=form
 */
function request({url, data, method="GET", contentType=1}) {
	let header = {
		'content-type': contentType === 1 ? 'application/json' : 'application/x-www-form-urlencoded'
	}
	let baseUrl = "http://localhost:3000";
	
	return new Promise((resolve, reject)=>{
		uni.request({
			url: baseUrl + url,
			data,
			method,
			header,
			success: (res) => {
				if (res.statusCode === 200) {
					//请求成功
					resolve(res.data);
				} else if (res.statusCode === 401) {
					uni.showToast({
						icon: 'none',
						title: "未登录或登录状态已超时",
						duration: 1500
					});
				} else if (res.statusCode === 405) {
					uni.showToast({
						icon: 'none',
						title: "请求方法错误",
						duration: 1500
					});
				} else {
					uni.showToast({
						icon: 'none',
						title: "请求错误:" + res.statusCode,
						duration: 1500
					});
				}
			},
			fail: (err) => {
				console.log("err:", err)
				uni.showToast({
					icon: 'none',
					title: err.errMsg,
					duration: 1500
				});
				reject(err);
			}
		})
	})
}
export default {request}

然后我们可以新建api文件夹新建index.js文件在里面进行api的封装
在这里插入图片描述

import request from "@/utils/request.js"

//轮播请求接口
export function apiGetBanner(data) {
	return request.request({
		url: '/banner',
		method: 'GET',
		data
	})
}

封装之后,在我们需要的页面就可以直接引入api进行使用,例如:

<script>
	import {apiGetBanner} from '@/apis/index.js',
	data() {
			return {
				swiper: [],                    //轮播
			}
		},
	methods: {
			getBanner() {
				apiGetBanner().then(res => {
					this.swiper = res.banners;
				})
			},
	}
</script>

主入口功能的实现:

下面来实现这一块的内容,首页的主入口部分
在这里插入图片描述
这一块的内容基本不会改动,所以直接用图片样式进行编写
这主要是把本地的图片渲染出来,css部分直接用flex布局拿很简单。

<view class="main-bar flex-box">
	<view class="flex-item" v-for="(item,index) in contentBar" :key="index">
		<image :src="'../../static/image/index/t_'+(index+1)+'.png'" class="img"></image>
		<view>
			{{item.name}}
		</view>
	</view>
</view>

推荐歌单模块

推荐歌单这块直接抽离成一个组件进行编写
在这里插入图片描述
在components文件夹下新建文件,编写歌单的组件

在这里插入图片描述
css样式我就不贴了,主要是写好组件,然后定义props接受父组件传过来的值

<template>
	<view class="song-list-comp">
		<view class="tit-bar">
			{{title}}
			<navigator :url="link" class="more fr">
				歌单广场
			</navigator>
		</view>
		<scroll-view class="scroll-view" scroll-x>
			<view class="item" v-for="(item,index) in list" :key="index">
				<image class="img" :src="item.picUrl" mode=""></image>
				<view class="desc ellipsis">
					{{item.name}}
				</view>
				<view class="count">
					{{item.playCount}}
				</view>
			</view>
		</scroll-view>
	</view>
</template>

<script>
	export default {
		props: {
			title: {
				type: String,
				default: ''
			},
			link: {
				type: String,
				default: ''
			},
			list: {
				type: Array,
				default: []
			}
		}
	}
</script>

在父组件中:

<song-list title="推荐歌单" link="" :list="recommendSongs"></song-list>

进行数据的请求,这里需要把请求数据中的播放量进行一个格式化

//推荐歌单
getRecommendSongs(){
	const params = {
		limit: 6
	}
	apiGetRecommendSongs(params).then(res => {
		//格式化播放量数据
		const formatCount = data=>{
			let tmp = data;
			if (data > 10000) {
				tmp = (parseInt(data/10000) + '万');
			}
			return tmp
		}
		this.recommendSongs = res.result;
		//格式化
		this.recommendSongs.forEach(item => {
			item.playCount = formatCount(item.playCount);
		})
	})
},

新碟新歌模块

这一块要实现一个可以点击切换的效果:
在这里插入图片描述

<!-- 新碟新歌 -->
<view class="song-list">
	<view class="switch-line flex-box">
			<view class="flex-box">
				<view class="switch-item" :class="{on : newType==1}" @click="switchTab(1)">
					新碟
				</view>
				<view class="switch-item" :class="{on : newType==2}" @click="switchTab(2)">
					新歌
				</view>
			</view>
			<template v-if="newType==1">
				<view class="more">
					更多新碟
				</view>
			</template>
			<template v-if="newType==2">
				<view class="more">
					更多新歌
				</view>
			</template>
	</view>
	<scroll-view class="scroll-view" scroll-x>
		<view class="item" v-for="(item,index) in latestAlbum" :key="index">
			<image class="img" :src="item.picUrl"></image>
			<view class="desc ellipsis">
				{{item.name}}
			</view>
			<view class="desc ellipsis c9">
				{{item.artist.name}}
			</view>
		</view>
	</scroll-view>
</view>

这里就是通过点击切换判断是新歌还是新碟,然后进行请求数据

// 切换新碟新歌
switchTab(type) {
	this.newType = type;
	// 设定开始start和结束end的位置
	let temp = {
		s: type == 1 ? 0 : 3,                  
		e: type == 1 ? 3 : 6
	}
	this.latestAlbum = this.latestTempAlbum.slice(temp.s, temp.e);
},
//新碟新歌  把前3首歌--新碟  后3首--新歌
getLatestAlbum() {
	apiGetTopAlbum().then(res=>{
		//将所有的数据暂存在临时变量中
		this.latestTempAlbum = res.albums;
		//取前3个作为第一类的数据展示
		this.latestAlbum = res.albums.slice(0, 3);
	})
},

精选视频模块

视频这款暂时做了封面,没有做视频播放功能,直接渲染封面:
在这里插入图片描述

<!-- 精选视频 -->
<view class="video-list song-list">
	<view class="tit-bar">
		精选视频
		<view class="more fr">
			更多
		</view>
	</view>
	<view class="video-item" v-for="(item, index) in relatedVideo" :key="index">
		<image :src="item.coverUrl" mode="" class="img"></image>
		<view class="desc ellipsis">
			{{item.title}}
		</view>
	</view>
</view>
//精选视频
			getRelatedVideo() {
				const params = {
					id: 32154      //根据资源id查询
				}
				apiGetRelatedVideo(params).then(res=>{
					// console.log(res)
					this.relatedVideo = res.data;
				})
			}

最后

到此首页基本实现了
在这里插入图片描述
后续会继续完成其他页面,持续更新,感兴趣可以订阅本专栏

在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

(详细及解决方法)关于Vue.prototype中定义的变量不是响应式

目录 一、背景 二、原因 三、解决方法 四、扩展 当时第一眼看到下面的图&#xff0c;就想这个不会是写错了吧&#xff0c;咋还能这样写&#xff0c;在这里我承认&#xff0c;我以前说话确实很大声了 一、背景 在项目中需要将全局变量存放到Vue的实例对象上面 Vue.protot…

【历史上的今天】12 月 13 日:时代杂志将“你”评为年度人物;苹果发布 AirPods;互联网传出欧洲

整理 | 王启隆 透过「历史上的今天」&#xff0c;从过去看未来&#xff0c;从现在亦可以改变未来。 今天是 2022 年 12 月 13 日&#xff0c;国家公祭日。从 2014 年开始&#xff0c;12 月 13 日被定为缅怀先烈的国家公祭日&#xff0c;网站也会在这一天变为全灰色&#xff0c…

墙裂推荐c++的学习网站(OJ)

目录 墙裂推荐OJ 墙裂推荐学习软件 墙裂推荐线上编辑器 墙裂推荐OJ 要想学好c,更好的OJ少不了啊!!! 为了让诸多小白们有更好的学习测试环境,特此准备了一下几个OJ, 1.Openjudge 网站: OpenJudge openjudge网站中储存着大量的题目,而且还提前帮你们把板块分好了,就问你们…

[Java] HashMap是如何实现的?扩容机制是什么?树化机制知道吗?结合源码带你理解HashMap的原理。

文章目录前言HashMap是什么&#xff1f;Map接口散列表&#xff08;HashTable&#xff09;HashMap的扩容机制扩容机制&#xff1f;扩谁的容&#xff1f;HashMap的容量&#xff08;Capacity&#xff09;属性HashMap的扩容因子&#xff08;load factor&#xff09;属性HashMap的树…

【js逆向基础】crypto 之 hash和hmac

▒ 目录 ▒&#x1f6eb; 导读需求开发环境1️⃣ hash对象创建Hash常见方法例封装成 stream 实例封装成管道流&#xff08;piped stream)2️⃣ hmac对象创建Hmac常见方法例封装成 stream 实例封装成管道流&#xff08;piped stream)&#x1f6ec; 文章小结&#x1f4d6; 参考资…

mysql学习---流程控制函数(条件判断)case when ,if使用

文章目录流程控制函数流程控制函数的介绍流程函数的使用流程控制函数 流程控制函数的介绍 if(value,value1,value2) 解释&#xff1a;如果value的结果为true &#xff0c;返回value1,否则返回value2 例如&#xff1a; select if(1>0,正确,错误) ->正确ifnull(value1,val…

Babel快速入门

一、简介 Babel是一个广泛使用的转码器&#xff0c;可以将ES6代码转为ES5代码&#xff0c;从而在现有环境执行执行。 这意味着&#xff0c;你可以现在就用 ES6 编写程序&#xff0c;而不用担心现有环境是否支持。 二、安装 安装命令行转码工具 Babel提供babel-cli工具&#xff…

Mysql调优(三)——通过索引进行优化、 查询优化

接上&#xff1a;Mysql调优&#xff08;二&#xff09;——schema与数据类型优化、执行计划 本篇文章在一定程度上与下面两篇文章重合&#xff0c;笔者建议优先读下面两篇文章&#xff0c;本文偏向概念&#xff0c;而它们偏向原理&#xff1a; Mysql索引的本质深入浅出&#x…

全栈Jmeter接口测试(五):json提取器元件及jsonpath介绍,响应断言元件

Jmeter(10)&#xff1a;json提取器元件及jsonpath介绍 json提取器元件介绍&#xff1a; json提取器与正则表达式提取器功能类似&#xff0c;也是用来截取响应信息的部分保 存到指定的变量中去&#xff0c;不同的是&#xff0c;它只能用来处理响应正文&#xff0c;并且响应正文…

ITE Super IO 学习 – 串口

串口都算是一个比较古老的功能了,也就是比较成熟了。Super IO中的大部分是硬件已经做好的,BIOS这边只需做小小的改动甚至默认便可以工作了。 串口寄存器 串口有关的寄存器总的来说有4个,两个用来设置IO资源的地址,一个是中断号选择,还有一个特殊配置寄存器。 Seri…

高级路由期末命令配置

文章目录前提要求(1)拓扑图搭建及IP地址规划拓扑图配置&#xff1a;IP地址表规划表1网络连接规划表表2 网络设备明细表表3 IP规划表&#xff08;2&#xff09;OSPF&#xff08;3&#xff09;BGP&#xff08;4&#xff09;路由引入&#xff08;5&#xff09;路由选择&#xff08…

PicoDet代码学习记录

推理步骤 [picodet_s_320_coco.yml] Infer.py main()->run()Trainer.py __init__() self.model create(cfg.architecture) Picodet.py from_config()->__init__() head create(cfg[head], **kwargs)时候调用&#xff1a; Layers.py MultiClassNMS __…

Django(一)

一、Django介绍 1、python语言编写的开源web框架 2、重量级python web框架&#xff0c;配备了常用的大部分组件 3、组件&#xff1a; ※基本配置文件/路由系统 ※模型层&#xff08;M&#xff09;/模板层&#xff08;T&#xff09;/视图层&#xff08;V&#xff09; ※co…

Cadence创建插件焊盘(超详细)

背景&#xff1a;Cadence画封装&#xff0c;在我看来&#xff0c;操作是比较不方便的事儿&#xff0c;所以在此在记录一遍。要画一个插件封装&#xff0c;需要调用画好的焊盘&#xff08;AD里直接设置即可&#xff09;&#xff0c;另外放置好pin的位置和绘制好边框。包括遇到的…

SpringBoot集成阿里EasyExcel导出excel高级实战

目录参考一、引入包二、导出到文件并输出到后台三、过滤字段方式1&#xff1a;类上加注解 ExcelIgnoreUnannotated&#xff0c;过滤属性没有ExcelProperty注解的字段方式2&#xff1a;指定字段加注解方式3&#xff1a;代码指定过滤字段, 同一个excel生成两个sheet分别过滤不同字…

clickhouse集群搭建

一、clickhouse单机部署 clickhouse镜像下载地址 https://mirrors.tuna.tsinghua.edu.cn/clickhouse/ 1、关闭防火墙 #查看防火墙状态systemctl status firewalld#关闭防火墙systemctl stop firewalld#设置防火墙服务开机不自启systemctl disable firewalld#再次查看防火墙检…

开源软件如何使企业和业务受益

在当今技术驱动、快速发展的商业环境中&#xff0c;越来越多的公司选择投资开源软件。开源为企业提供了许多优势&#xff1a;更快的创新步伐、来自庞大而热情的社区的强大支持以及不受供应商锁定的影响。 对于许多人来说&#xff0c;开源不仅仅是编码。它代表了一种生活方式。…

使用IDEA工具,通过Java API 操作 HDFS (文件/目录的操作,含源码,详细操作步骤)

文章目录一&#xff0c;了解 HDFS Java API&#xff08;一&#xff09;HDFS常见类与接口&#xff08;二&#xff09;FileSystem的常用方法二&#xff0c;编写Java程序访问HDFS01 创建Maven项目02 添加相关依赖03 创建日志属性文件&#xff08;1&#xff09;在resources目录里创…

2022年还剩半个月,我从外包公司离职了...

今天是12.12&#xff0c;距离元旦也仅剩半个月&#xff0c;大家都在考论假期怎么过&#xff0c;没错&#xff0c;我离职了... 19年大专毕业通过校招进入了一家外包公司&#xff0c;干了接近3年的功能测试&#xff0c;今年年初&#xff0c;感觉自己不能够在这样下去了&#xff…

MyBatis 注解开发

文章目录一、单表查询1. Select 注解2. Insert 注解3. Update 注解4. Delete 注解5. Param 注解二、多表关联查询1. 一对一查询2. 一对多查询3. 多对多查询一、单表查询 1. Select 注解 前面的章节介绍了 MyBatis 的基本用法、关联映射、动态 SQL 和缓存机制等知识&#xff0…