Kotlin项目实战之手机影音---加载mv界面区域数据、mv界面viewpager适配、tablayout适配、mv每一个界面列表显示

本文主要是介绍Kotlin项目实战之手机影音---加载mv界面区域数据、mv界面viewpager适配、tablayout适配、mv每一个界面列表显示,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

修复API:

距离上一次Kotlin项目实战之手机影音---悦单条目实现及BaseListFragment抽取的这个Kotlin项目的学习间隔一年多了,完成已经消失在自己的记忆当中了,当然不能半途而废,继续接着前行,然后运行把工程重新运行一下,发现项目起不来了,尴尬。。

原因是由于音乐的API的域名访问不了了。。其实也正常,因为在之前Kotlin项目实战之手机影音---主界面tab切换、home界面适配、获得首页网络数据就说音乐数据是从网上找的,比如进首页的api为:

然后到浏览器访问一下:

寻找新的网上音乐API:

只能另找一些免费的数据源了【如果之后又被封了,我打算自己搭建个后台mock数据得了,要不用node服务器,要不用springboot,总之办法大于困难~~】,在网上搜到个这个免费稳定 => 网易云音乐API_余光的博客-CSDN博客_免费音乐api,打开:

然后到github上GitHub - Binaryify/NeteaseCloudMusicApi: 网易云音乐 Node.js API service把源码下来,按官方步骤安装运行一下:

然后输入“http://localhost:3000/”,此时会打开如下页面:

然后点击“查看文档”,此时可以看到列了一大堆接口,那哪个能为咱们项目所用呢?找来找去,没有发现特别适合的,不过这个勉强能用:

怎么用呢?这样访问:http://localhost:3000/mv/all?area=%E6%B8%AF%E5%8F%B0&limit=2&offset=0

那咱们这个项目是支持音频和视频的播放,这里获取的是mv列表,而具体mv的地址需要通过另一个接口获取,如下:

其中请求的id参数来自于它:

那。。有时候会有音频mp3,那咋搞呢?这里打算用一个写死的mp3地址作为列表中的一个实体元素就行了,完全不影响咱们的学习。

修改API:

接下来,则来修改一下项目的API,目前已实现的页面中涉及到两个URL,如下:

package com.kotlin.musicplayer.utils;import android.util.Log;public class URLProviderUtils {/*** 获取首页的url** @param offset 数据偏移量* @param size   返回数据的条目个数* @return url*/public static String getHomeUrl(int offset, int size) {String url = "http://tingapi.ting.baidu.com/v1/restserver/ting?from=android&version=5.6.5.0&method=baidu.ting.artist.getSongList&format=json&order=2&tinguid=7988&artistid=7988"+ "&offset=" + offset+ "&limits=" + size;Log.i("Main_url", url);return url;}/*** 获取悦单列表的url*/public static String getYueDanUrl(int offset, int size) {String url = "http://tingapi.ting.baidu.com/v1/restserver/ting?from=android&version=5.6.5.0&method=baidu.ting.artist.getSongList&format=json&order=2&tinguid=7988&artistid=7988"+ "&offset=" + offset+ "&limits=" + size;Log.i("Main_url", url);return url;}
}

这里修改成网易云音乐的API:

然后修改对应的实体,如下:

package com.kotlin.musicplayer.model/*** home界面每个条目的bean*/
data class HomeBean(val data: List<HomeItemBean>
)data class HomeItemBean(val artistId: Int,val artistName: String,val briefDesc: Any,val cover: String,val desc: Any,val duration: Int,val id: Int,val mark: Int,val name: String,val playCount: Int,val subed: Boolean
)
package com.kotlin.musicplayer.model/*** 悦单列表bean*/
data class YueDanBean(val data: List<YueDanItemBean>
)data class YueDanItemBean(val artistId: Int,val artistName: String,val briefDesc: Any,val cover: String,val desc: Any,val duration: Int,val id: Int,val mark: Int,val name: String,val playCount: Int,val subed: Boolean
)

其中这两个界面都是需要有mp3音频地址的对吧,所以这里各增加一个字段,然后里面写死一个在线的mp3地址:

package com.kotlin.musicplayer.model/*** home界面每个条目的bean*/
data class HomeBean(val data: List<HomeItemBean>
)data class HomeItemBean(val artistId: Int,val artistName: String,val briefDesc: Any,val cover: String,val desc: Any,val duration: Int,val id: Int,val mark: Int,val name: String,val playCount: Int,val subed: Boolean,val audioUrl: String = "http://m8.music.126.net/20210817054609/f3b6ded4f4a9321b6a26d26cfacc135f/ymusic/obj/w5zDlMODwrDDiGjCn8Ky/3047366729/c22a/6e45/ff05/31dbb8ef2bffa556d44aa24306e0ce1f.mp3"
)
package com.kotlin.musicplayer.model/*** 悦单列表bean*/
data class YueDanBean(val data: List<YueDanItemBean>
)data class YueDanItemBean(val artistId: Int,val artistName: String,val briefDesc: Any,val cover: String,val desc: Any,val duration: Int,val id: Int,val mark: Int,val name: String,val playCount: Int,val subed: Boolean,val audioUrl: String = "http://m8.music.126.net/20210817054609/f3b6ded4f4a9321b6a26d26cfacc135f/ymusic/obj/w5zDlMODwrDDiGjCn8Ky/3047366729/c22a/6e45/ff05/31dbb8ef2bffa556d44aa24306e0ce1f.mp3"
)

其中audioUrl的这个歌曲也是通过这个网易API来获取的,真的挺方便的,我想要林俊杰-曹操这首歌,然后先找到这位歌手的id:

然后再通过这个API来获取这位歌手的所有歌曲,其中找到想要歌曲的id信息:

然后再根据这个id来用另一个API获取具体的mp3地址:

另外注意,这里的字段重命名了:

修改Adapter:

由于model的字段变了,列表的显示也得进行调整,如下:

运行:

ok,环境恢复了,接下来可以继续前行了。

加载mv界面区域数据:

效果:

搭建布局:

对于这种滑动效果可以使用Indicator+ViewPager来实现,但是在Android5.0之后提供了一个TabLayout组件,咱们这里来使用它,如下:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"xmlns:app="http://schemas.android.com/apk/res-auto"android:layout_width="match_parent"android:layout_height="match_parent"android:orientation="vertical"><com.google.android.material.tabs.TabLayoutandroid:id="@+id/tabLayout"android:layout_width="match_parent"android:layout_height="wrap_content"android:background="@color/colorPrimary"app:tabIndicatorColor="#fff"app:tabIndicatorHeight="2dp"app:tabMode="scrollable"app:tabSelectedTextColor="#fff"app:tabTextColor="#40000000" /><androidx.viewpager.widget.ViewPagerandroid:id="@+id/viewPager"android:layout_width="match_parent"android:layout_height="match_parent" />
</LinearLayout>

加载tab音乐分类数据:

很显然对于上面的tab是从网络上来获取的,这边正好也有现成的接口:

创建MVView接口:

该接口是用来实现P层数据回调给UI层的,这里先新建一下,里面的接口方法之后用到时再来定义:

package com.kotlin.musicplayer.viewimport com.kotlin.musicplayer.model.MVTaginterface MVView {fun onError(msg: String?)fun onSuccess(result: List<MVTag>)
}

其中这里不像之前的页面一样去继承BaseView了:

因为没有下拉刷新列表动作,然后涉及到一个model:

package com.kotlin.musicplayer.modeldata class MvAreaBean(val tags: List<MVTag>
)data class MVTag(val category: Int,val hot: Boolean,val id: Int,val name: String,val type: Int
)

创建P层:

package com.kotlin.musicplayer.presenter.`interface`interface MvPresenter {fun loadDatas()
}
package com.kotlin.musicplayer.presenter.implimport com.kotlin.musicplayer.model.MvAreaBean
import com.kotlin.musicplayer.net.ResponseHandler
import com.kotlin.musicplayer.presenter.`interface`.MvPresenter
import com.kotlin.musicplayer.view.MVViewclass MvPresenterImpl(var mvView: MVView) : MvPresenter, ResponseHandler<MvAreaBean> {override fun loadDatas() {}override fun onError(type: Int, msg: String?) {TODO("Not yet implemented")}override fun onSuccessed(type: Int, result: MvAreaBean) {TODO("Not yet implemented")}
}

界面中调用P:

实现数据加载的逻辑:

package com.kotlin.musicplayer.netimport com.kotlin.musicplayer.model.MvAreaBean
import com.kotlin.musicplayer.utils.URLProviderUtils/*** mv区域数据请求*/
class MvAreaRequest(handler: ResponseHandler<MvAreaBean>) :MRequest<MvAreaBean>(0, URLProviderUtils.getMVareaUrl(), handler) {
}

然后在界面简单处理一下结果回调,目前先不处理列表显示,只先打个toast出来,先确保接口是畅通的:

运行:

mv界面viewpager适配:

准备PagerAdapter:

package com.kotlin.musicplayer.adapterimport android.os.Bundle
import androidx.fragment.app.Fragment
import androidx.fragment.app.FragmentManager
import androidx.fragment.app.FragmentPagerAdapter
import com.kotlin.musicplayer.model.MVTag
import com.kotlin.musicplayer.ui.fragment.MvPagerFragment/*** MV PagerAdapter*/
class MvPagerAdapter(val list: List<MVTag>?, fm: FragmentManager?) :FragmentPagerAdapter(fm) {override fun getItem(position: Int): Fragment {//第一种数据传递方式val fragment = MvPagerFragment()val bundle = Bundle()bundle.putString("args", list?.get(position)?.name)fragment.arguments = bundlereturn fragment}override fun getCount(): Int {return list?.size ?: 0//如果不为null返回list.size 为空返回0}override fun getPageTitle(position: Int): CharSequence? {return list?.get(position)?.name}
}

其中要注意的是getPageTitile,指的是Indicator的显示文案。接下来准备Fragment:

绑定Adapter:

接下来则回到界面上当数据请求成功之后绑定一下Adapter,如下:

运行:

然后运行看一下效果:

mv每一个界面列表显示:

接下来则来处理每一个Fragment的列表的显示处理了,由于我们之前已经对于列表的显示已经做了base的封装,所以说,实现起来也非常之容易。

准备Item bean:

package com.kotlin.musicplayer.modeldata class MvPagerBean(val data: List<VideosBean>
)data class VideosBean(val artistId: Int,val artistName: String,val briefDesc: Any,val cover: String,val desc: Any,val duration: Int,val id: Int,val mark: Int,val name: String,val playCount: Int,val subed: Boolean,val videoUrl: String = "http://vodkgeyttp8.vod.126.net/cloudmusic/obj/core/9337503943/e78b28855251f5c900771871537e6ae4.mp4?wsSecret=df2c6d3b9146c33d48b18840923a27f5&wsTime=1629178848"
)

其中MV地址也是写死了,来自林俊杰-裹着心的光。

准备Item View:

package com.kotlin.musicplayer.widgetimport android.content.Context
import android.util.AttributeSet
import android.view.View
import android.widget.RelativeLayout
import com.itheima.player.model.bean.VideosBean
import com.kotlin.musicplayer.R
import com.squareup.picasso.Picasso
import kotlinx.android.synthetic.main.item_mv.view.*/*** mv每一个界面条目view*/
class MvItemView : RelativeLayout {constructor(context: Context?) : super(context)constructor(context: Context?, attrs: AttributeSet?) : super(context, attrs)constructor(context: Context?, attrs: AttributeSet?, defStyleAttr: Int) : super(context,attrs,defStyleAttr)init {View.inflate(context, R.layout.item_mv, this)}/*** 适配每一个条目view*/fun setData(data: VideosBean) {//歌手名称artist.text = data.artistName//歌曲名称title.text = data.name//背景图Picasso.get().load(data.cover).into(bg)}
}

布局文件:

<?xml version="1.0" encoding="utf-8"?>
<androidx.cardview.widget.CardView xmlns:android="http://schemas.android.com/apk/res/android"xmlns:app="http://schemas.android.com/apk/res-auto"android:layout_width="match_parent"android:layout_height="wrap_content"app:cardCornerRadius="5dp"app:cardElevation="5dp"app:cardUseCompatPadding="true"><RelativeLayoutandroid:layout_width="match_parent"android:layout_height="300dp"android:padding="10dp"><ImageViewandroid:id="@+id/bg"android:layout_width="match_parent"android:layout_height="match_parent"android:scaleType="centerCrop" /><TextViewandroid:id="@+id/title"android:layout_width="wrap_content"android:layout_height="wrap_content"android:layout_alignParentBottom="true"android:maxLines="1"android:text="歌曲名称"android:textColor="#fff"android:textSize="20sp" /><TextViewandroid:id="@+id/artist"android:layout_width="wrap_content"android:layout_height="wrap_content"android:layout_above="@id/title"android:maxLines="1"android:text="歌手名称"android:textColor="#00ff00"android:textSize="25sp" /></RelativeLayout>
</androidx.cardview.widget.CardView>

准备列表Adapter:

package com.kotlin.musicplayer.adapterimport android.content.Context
import com.kotlin.musicplayer.base.BaseListAdapter
import com.kotlin.musicplayer.model.VideosBean
import com.kotlin.musicplayer.widget.MvItemView/*** mv界面每一个list列表的适配器*/
class MvListAdapter : BaseListAdapter<VideosBean, MvItemView>() {override fun refreshItemView(itemView: MvItemView, data: VideosBean) {itemView.setData(data)}override fun getItemView(context: Context?): MvItemView {return MvItemView(context)}
}

准备P、V层:

package com.kotlin.musicplayer.viewimport com.kotlin.musicplayer.base.BaseView
import com.kotlin.musicplayer.model.MvPagerBeaninterface MvListView : BaseView<MvPagerBean> {
}
package com.kotlin.musicplayer.presenter.`interface`import com.kotlin.musicplayer.base.BaseListPresenterinterface MvListPresenter : BaseListPresenter {
}
package com.kotlin.musicplayer.presenter.implimport com.kotlin.musicplayer.model.MvPagerBean
import com.kotlin.musicplayer.net.ResponseHandler
import com.kotlin.musicplayer.presenter.`interface`.MvListPresenter
import com.kotlin.musicplayer.view.MvListViewclass MvListPresenterImpl(var code: String, var mvListView: MvListView?) : MvListPresenter,ResponseHandler<MvPagerBean> {override fun onError(type: Int, msg: String?) {mvListView?.onError(msg)}override fun loadDatas() {//TODO}override fun loadMoreDatas(offset: Int) {TODO("Not yet implemented")}override fun onSuccessed(type: Int, result: MvPagerBean) {TODO("Not yet implemented")}override fun destoryView() {if (mvListView != null) {mvListView = null}}
}

准备列表接口请求:

package com.kotlin.musicplayer.netimport com.kotlin.musicplayer.model.MvPagerBean
import com.kotlin.musicplayer.utils.URLProviderUtils/*** mv每一个界面数据网络请求*/
class MvListRequest(type: Int, code: String, offset: Int, handler: ResponseHandler<MvPagerBean>) :MRequest<MvPagerBean>(type, URLProviderUtils.getMVListUrl(code, offset, 5), handler) {
}

其中url为:

实现MvListPresenterImpl:

package com.kotlin.musicplayer.presenter.implimport com.kotlin.musicplayer.base.BaseListPresenter
import com.kotlin.musicplayer.model.MvPagerBean
import com.kotlin.musicplayer.net.MvListRequest
import com.kotlin.musicplayer.net.ResponseHandler
import com.kotlin.musicplayer.presenter.`interface`.MvListPresenter
import com.kotlin.musicplayer.view.MvListViewclass MvListPresenterImpl(var code: String, var mvListView: MvListView?) : MvListPresenter,ResponseHandler<MvPagerBean> {override fun onError(type: Int, msg: String?) {mvListView?.onError(msg)}override fun loadDatas() {MvListRequest(BaseListPresenter.TYPE_INIT_OR_REFRESH, code, 0, this).excute()}override fun loadMoreDatas(offset: Int) {MvListRequest(BaseListPresenter.TYPE_LOAD_MORE, code, offset, this).excute()}override fun destoryView() {if (mvListView != null) {mvListView = null}}override fun onSuccessed(type: Int, result: MvPagerBean) {if (type == BaseListPresenter.TYPE_INIT_OR_REFRESH) {mvListView?.loadSuccessed(result)} else if (type == BaseListPresenter.TYPE_LOAD_MORE) {mvListView?.onLoadMore(result)}}
}

实现MvPagerFragment:

package com.kotlin.musicplayer.ui.fragmentimport com.kotlin.musicplayer.adapter.MvListAdapter
import com.kotlin.musicplayer.base.BaseListAdapter
import com.kotlin.musicplayer.base.BaseListFragment
import com.kotlin.musicplayer.base.BaseListPresenter
import com.kotlin.musicplayer.model.MvPagerBean
import com.kotlin.musicplayer.model.VideosBean
import com.kotlin.musicplayer.presenter.impl.MvListPresenterImpl
import com.kotlin.musicplayer.view.MvListView
import com.kotlin.musicplayer.widget.MvItemView/*** mv界面每一个页面fragment*/
class MvPagerFragment : BaseListFragment<MvPagerBean, VideosBean, MvItemView>(), MvListView {var tinguid: String? = nulloverride fun init() {tinguid = arguments?.getString("args")}override fun getSpecialAdapter(): BaseListAdapter<VideosBean, MvItemView> {return MvListAdapter()}override fun getSpecialPresenter(): BaseListPresenter {return MvListPresenterImpl(tinguid!!, this)}override fun getList(response: MvPagerBean?): List<VideosBean>? {return response?.songlist}}

运行:

实测过程中有可能会遇到接口返回500的情况:

这个没办法,人家服务器的事,咱们这边加上加载失败的处理就成,如下:

关注个人公众号,获得实时推送

 

这篇关于Kotlin项目实战之手机影音---加载mv界面区域数据、mv界面viewpager适配、tablayout适配、mv每一个界面列表显示的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Python开发文字版随机事件游戏的项目实例

《Python开发文字版随机事件游戏的项目实例》随机事件游戏是一种通过生成不可预测的事件来增强游戏体验的类型,在这篇博文中,我们将使用Python开发一款文字版随机事件游戏,通过这个项目,读者不仅能够... 目录项目概述2.1 游戏概念2.2 游戏特色2.3 目标玩家群体技术选择与环境准备3.1 开发环境3

Java注解之超越Javadoc的元数据利器详解

《Java注解之超越Javadoc的元数据利器详解》本文将深入探讨Java注解的定义、类型、内置注解、自定义注解、保留策略、实际应用场景及最佳实践,无论是初学者还是资深开发者,都能通过本文了解如何利用... 目录什么是注解?注解的类型内置注编程解自定义注解注解的保留策略实际用例最佳实践总结在 Java 编程

电脑显示mfc100u.dll丢失怎么办?系统报错mfc90u.dll丢失5种修复方案

《电脑显示mfc100u.dll丢失怎么办?系统报错mfc90u.dll丢失5种修复方案》最近有不少兄弟反映,电脑突然弹出“mfc100u.dll已加载,但找不到入口点”的错误提示,导致一些程序无法正... 在计算机使用过程中,我们经常会遇到一些错误提示,其中最常见的就是“找不到指定的模块”或“缺少某个DL

一文教你Python如何快速精准抓取网页数据

《一文教你Python如何快速精准抓取网页数据》这篇文章主要为大家详细介绍了如何利用Python实现快速精准抓取网页数据,文中的示例代码简洁易懂,具有一定的借鉴价值,有需要的小伙伴可以了解下... 目录1. 准备工作2. 基础爬虫实现3. 高级功能扩展3.1 抓取文章详情3.2 保存数据到文件4. 完整示例

使用Java将各种数据写入Excel表格的操作示例

《使用Java将各种数据写入Excel表格的操作示例》在数据处理与管理领域,Excel凭借其强大的功能和广泛的应用,成为了数据存储与展示的重要工具,在Java开发过程中,常常需要将不同类型的数据,本文... 目录前言安装免费Java库1. 写入文本、或数值到 Excel单元格2. 写入数组到 Excel表格

python处理带有时区的日期和时间数据

《python处理带有时区的日期和时间数据》这篇文章主要为大家详细介绍了如何在Python中使用pytz库处理时区信息,包括获取当前UTC时间,转换为特定时区等,有需要的小伙伴可以参考一下... 目录时区基本信息python datetime使用timezonepandas处理时区数据知识延展时区基本信息

Qt实现网络数据解析的方法总结

《Qt实现网络数据解析的方法总结》在Qt中解析网络数据通常涉及接收原始字节流,并将其转换为有意义的应用层数据,这篇文章为大家介绍了详细步骤和示例,感兴趣的小伙伴可以了解下... 目录1. 网络数据接收2. 缓冲区管理(处理粘包/拆包)3. 常见数据格式解析3.1 jsON解析3.2 XML解析3.3 自定义

SpringMVC 通过ajax 前后端数据交互的实现方法

《SpringMVC通过ajax前后端数据交互的实现方法》:本文主要介绍SpringMVC通过ajax前后端数据交互的实现方法,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价... 在前端的开发过程中,经常在html页面通过AJAX进行前后端数据的交互,SpringMVC的controll

SpringBoot中配置文件的加载顺序解读

《SpringBoot中配置文件的加载顺序解读》:本文主要介绍SpringBoot中配置文件的加载顺序,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录SpringBoot配置文件的加载顺序1、命令⾏参数2、Java系统属性3、操作系统环境变量5、项目【外部】的ap

Pandas统计每行数据中的空值的方法示例

《Pandas统计每行数据中的空值的方法示例》处理缺失数据(NaN值)是一个非常常见的问题,本文主要介绍了Pandas统计每行数据中的空值的方法示例,具有一定的参考价值,感兴趣的可以了解一下... 目录什么是空值?为什么要统计空值?准备工作创建示例数据统计每行空值数量进一步分析www.chinasem.cn处