前言
音乐播放器在目前手机应用中非常普遍,安卓开发中对于音乐盒的实现方法进行掌握也十分重要,今天我将总结一下播放的实现过程,有兴趣的朋友可以参考借鉴一下思路。
开发工具
Android Studio
开发环境
1)JDK(Java Development Kit)JDK是Java语言的软件开发工具包,主要用于移动设备、嵌入设备上的java应用程序。
2)SDK(software development kit) SDK是软件开发工具包。 被软件开发工程师用于为特定的软件包、软件框架、硬件平台、操作系统等建立应用软件的开发工具的集合
实现过程
在我看来,每个完整的项目都要分UI和逻辑处理两部分。UI的美观固然重要,但是也要注意实现过程的复杂程度与可行性。
界面设计:
界面上:设计了上一首、暂停(开始)、下一首三个按钮,还有一个Listview用于展现歌曲
代码如下:
<?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"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context="com.zewei_w.mp3player.MainActivity"
android:orientation="vertical">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="126dp"
android:orientation="horizontal">
<TextView
android:id="@+id/position"
android:layout_width="120dp"
android:layout_height="match_parent"
android:layout_alignParentBottom="true"
android:layout_alignParentEnd="true"
android:layout_alignParentStart="true"
android:background="@mipmap/bg" />
<RelativeLayout
android:layout_width="203dp"
android:layout_height="match_parent">
<Button
android:id="@+id/last"
android:layout_width="60dp"
android:layout_marginLeft="1dp"
android:layout_height="60dp"
android:layout_gravity="center_vertical"
android:background="@mipmap/last"
/>
<Button
android:id="@+id/start"
android:layout_width="60dp"
android:layout_height="60dp"
android:layout_gravity="center_vertical"
android:layout_marginLeft="10dp"
android:layout_marginRight="10dp"
android:layout_toRightOf="@id/last"
android:background="@mipmap/start" />
<Button
android:layout_width="60dp"
android:id="@+id/next"
android:layout_height="60dp"
android:layout_gravity="center_vertical"
android:background="@mipmap/next"
android:layout_alignParentTop="true"
android:layout_alignParentEnd="true"
android:layout_toEndOf="@+id/start" />
<TextView
android:id="@+id/songname"
android:layout_width="wrap_content"
android:layout_height="40dp"
android:layout_below="@id/start"
android:text="无"
android:textColor="@color/black"
android:textSize="15dp" />
<TextView
android:id="@+id/singer"
android:layout_width="wrap_content"
android:layout_height="20dp"
android:layout_below="@id/songname"
android:gravity="top"
android:text="无"
android:textSize="10dp" />
/>
</RelativeLayout>
</LinearLayout>
<TextView
android:layout_width="match_parent"
android:layout_height="30dp"
android:text="本地音乐"
android:textSize="20dp"
android:textColor="@color/white"
android:background="@color/black0"/>
<ListView
android:id="@+id/listview"
android:layout_width="match_parent"
android:layout_height="match_parent"
>
</ListView>
</LinearLayout>
效果如图:
需要注意的是ListView中每一项又是一个单独的布局,因此还需要设计一个布局供其引入,主要是一个Textview显示序列、一个TextView显示歌曲名、一个TextView显示歌手
代码如下:
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_height="60dp"
android:layout_width="match_parent">
<TextView
android:id="@+id/position"
android:layout_width="60dp"
android:layout_height="60dp"
android:gravity="center" />
<RelativeLayout
android:layout_width="match_parent"
android:layout_height="match_parent">
<TextView
android:id="@+id/songname"
android:layout_width="wrap_content"
android:layout_height="40dp"
android:textColor="@color/black"
android:text="歌曲名"
android:textSize="15dp" />
<TextView
android:id="@+id/singer"
android:gravity="top"
android:layout_width="wrap_content"
android:layout_height="20dp"
android:layout_below="@id/songname"
android:text="歌手"
android:textSize="10dp" />
/>
</RelativeLayout>
</LinearLayout>
其他部分
程序中有四个类:
MainActivity代码:
音频播放的实现是通过MediaPlayer实现的,实现方法为以下这几步:
mediaPlayer.reset();
//调用方法传进去要播放的音频路径
mediaPlayer.setDataSource(path);
//异步准备音频资源
mediaPlayer.prepareAsync();
//调用mediaPlayer的监听方法,音频准备完毕会响应此方法
mediaPlayer.setOnPreparedListener(new MediaPlayer.OnPreparedListener() {
@Override
public void onPrepared(MediaPlayer mediaPlayer) {
mediaPlayer.start();//开始音频
}
});
还运用了 mediaPlayer.setOnCompletionListener设置监听器,当音乐播放结束时自动播放下一首,或者其它的逻辑处理可在这里实现。
package com.zewei_w.mp3player;
import android.content.Context;
import android.media.MediaPlayer;
import android.provider.ContactsContract;
import android.provider.Settings;
import android.support.v4.text.TextDirectionHeuristicCompat;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.view.WindowManager;
import android.widget.AdapterView;
import android.widget.ImageButton;
import android.widget.Button;
import android.widget.ImageView;
import android.widget.ListView;
import android.widget.TextView;
import android.widget.Toast;
import java.io.IOException;
import java.util.ArrayList;
import java.util.List;
public class MainActivity extends AppCompatActivity implements View.OnClickListener{
private ListView listview;
private List<song> list;
private int playPosition=0;
private Button Button1;
private Button Button2;
private Button Button3;
private Myadapter myadapter;
private TextView songn;
private TextView singer;
private boolean isplay=false;
private MediaPlayer mediaPlayer=new MediaPlayer();
public MainActivity() {
}
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
listview = (ListView) findViewById(R.id.listview);
list = new ArrayList<>();
list = search.getMusicData(this);
myadapter = new Myadapter(this, list);
listview.setAdapter(myadapter);
songn=(TextView)findViewById(R.id.songname);
singer=(TextView)findViewById(R.id.singer);
Button1 =(Button)findViewById(R.id.last);
Button2 =(Button)findViewById(R.id.start);
Button3 =(Button)findViewById(R.id.next);
Button1.setOnClickListener(this);
Button2.setOnClickListener(this);
Button3.setOnClickListener(this);
listview.setOnItemClickListener(new AdapterView.OnItemClickListener() {
@Override
public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
play(list.get(position).path);
Button2.setBackgroundResource(R.mipmap.pause);
playPosition=position;
isplay=true;
setText(position);
}
});
}
private void setText(int Position)
{
int playPostion=Position;
songn.setText(list.get(playPostion).songname);
singer.setText(list.get(playPostion).singer);
}
private void play(String path) {
//播放之前要先把音频文件重置
try {
mediaPlayer.reset();
//调用方法传进去要播放的音频路径
mediaPlayer.setDataSource(path);
//异步准备音频资源
mediaPlayer.prepareAsync();
//调用mediaPlayer的监听方法,音频准备完毕会响应此方法
mediaPlayer.setOnPreparedListener(new MediaPlayer.OnPreparedListener() {
@Override
public void onPrepared(MediaPlayer mediaPlayer) {
mediaPlayer.start();//开始音频
}
});
} catch (IOException e) {
e.printStackTrace();
}
mediaPlayer.setOnCompletionListener(new MediaPlayer.OnCompletionListener() {
@Override
public void onCompletion(MediaPlayer mp) {
if(playPosition<(list.size()-1))
{
playPosition++;
play(list.get(playPosition).path);
setText(playPosition);
}
if(playPosition==(list.size()-1))
{ play(list.get(playPosition).path);
playPosition++;
mediaPlayer.pause();
Toast.makeText(MainActivity.this, "播放结束", Toast.LENGTH_SHORT).show();
Button2.setBackgroundResource(R.mipmap.start);
isplay=false;
}
}
});
}
@Override
public void onClick(View v) {
switch (v.getId())
{
case R.id.last:
if(playPosition<=0)
{
Toast.makeText(MainActivity.this,"已经是第一首歌",Toast.LENGTH_SHORT).show();
}
else
{
playPosition--;
play(list.get(playPosition).path);
Button2.setBackgroundResource(R.mipmap.start);
Button2.setBackgroundResource(R.mipmap.pause);
isplay=true;
setText(playPosition);
}
break;
case R.id.start:
if(isplay==true)
{
Button2.setBackgroundResource(R.mipmap.start);
isplay=false;
mediaPlayer.pause();
}
else
{
Button2.setBackgroundResource(R.mipmap.pause);
isplay=true;
mediaPlayer.start();
}
break;
case R.id.next:
if(playPosition>=(list.size()-1))
{
Toast.makeText(MainActivity.this,"已经是最后一首歌了",Toast.LENGTH_SHORT);
}
else
{
playPosition++;
play(list.get(playPosition).path);
Button2.setBackgroundResource(R.mipmap.pause);
isplay=true;
setText(playPosition);
}
break;
}
}
}
MyAdapter类:自定义适配器,用于将歌曲展现在ListView
代码:
package com.zewei_w.mp3player;
import android.content.Context;
import android.view.View;
import android.view.ViewGroup;
import android.widget.Adapter;
import android.widget.BaseAdapter;
import android.widget.ListView;
import android.widget.TextView;
import java.util.List;
/**
* Created by ZeWei-W on 2017/8/15.
*/
public class Myadapter extends BaseAdapter{
private Context context;
private List<song> list;
public Myadapter(Context activity, List<song> list1) {
context=activity;
list=list1;
}
public int getCount() {
return list.size();
}
@Override
public Object getItem(int position) {
return list.get(position);
}
@Override
public long getItemId(int position) {
return position;
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
ViewHolder viewHolder;
if (convertView == null) {
viewHolder = new ViewHolder();
convertView = View.inflate(context, R.layout.listview_layout, null);
viewHolder.songname = (TextView) convertView.findViewById(R.id.songname);
viewHolder.singer = (TextView) convertView.findViewById(R.id.singer);
viewHolder.position = (TextView) convertView.findViewById(R.id.position);
convertView.setTag(viewHolder);
}
viewHolder = (ViewHolder) convertView.getTag();
viewHolder.songname.setText(list.get(position).songname.toString());
viewHolder.singer.setText(list.get(position).singer.toString());
viewHolder.position.setText(position + 1 + "");
return convertView;
}
public class ViewHolder
{
public TextView songname;
public TextView singer;
public TextView position;
}
}
song类:
package com.zewei_w.mp3player;
/**
* Created by ZeWei-W on 2017/8/15.
*/
public class song {
public String songname;
public String singer;
public String path;
public int duration;//歌曲长度
public long size; //歌曲大小
}
search类:用于查找音乐文件
当我们往手机上放图片或者音乐的时候,会在手机内存中某个位置上的某个database中存放图片或者音乐的信息,而我们的应用程序是能够通过ContentResolver去读取到这些数据的
这里我用的是MediaStore.Audio.Media.INTERNAL_CONTENT_URI,只能读取到手机内存自带的一些音频,如铃声等,手机内存其他文件夹的音乐却扫描不到,这是笔者还未解决的一个问题,如果你用的是MediaStore.Audio.Media.EXTERNAL_CONTENT_URI这个URI,扫描的则是手机外部SD卡中的音乐文件。
代码如下:
package com.zewei_w.mp3player;
import android.content.Context;
import android.database.Cursor;
import android.database.sqlite.SQLiteDatabase;
import android.graphics.PorterDuff;
import android.net.Uri;
import android.os.Environment;
import android.provider.MediaStore;
import android.provider.Settings;
import android.text.Selection;
import java.util.ArrayList;
import java.util.Currency;
import java.util.List;
/**
* Created by ZeWei-W on 2017/8/15.
*/
public class search {
/**
* 扫描系统里面的音频文件,返回一个list集合
*/
public static List<song> getMusicData(Context context) {
List<song> list = new ArrayList<song>();
Cursor cursor = context.getContentResolver().query(MediaStore.Audio.Media.INTERNAL_CONTENT_URI, null,null,
null, MediaStore.Audio.AudioColumns.IS_MUSIC);
if (cursor != null) {
while (cursor.moveToNext()) {
song son = new song();
son.songname = cursor.getString(cursor.getColumnIndex(MediaStore.MediaColumns.TITLE));
son.singer = cursor.getString(cursor.getColumnIndexOrThrow(MediaStore.Audio.Media.ARTIST));
son.path = cursor.getString(cursor.getColumnIndexOrThrow(MediaStore.Audio.Media.DATA));
son.duration = cursor.getInt(cursor.getColumnIndexOrThrow(MediaStore.Audio.Media.DURATION));
son.size = cursor.getLong(cursor.getColumnIndexOrThrow(MediaStore.Audio.Media.SIZE));
/* if (son.size > 1000 * 800) {
if (son.songname.contains("-")) {
String[] str = son.songname.split("-");
son.singer = str[0];
son.songname = str[1];
}
}*/
list.add(son);
System.out.println(son.path);
}
cursor.close();
}
else
{
System.out.print("null");
}
return list;
}
}
可拓展处
1.实现搜索网络歌曲
2.实现扫描具体文件夹
3.实现歌曲歌词的展现
4.实现下载歌曲
个人总结:
通过这个小项目,我还是学到了不少的知识点,但是由于刚入门,其实做的非常简单,仅仅是实现了音频的播放,很多拓展功能并没有实现完全,并且在音乐的扫描这里也遇到了不少问题,接下来还应多加研究,将其总结在文章上,仅仅是作为一个记录,当然,如果能对正在学习的朋友有一丝丝的作用还是很高兴,更希望有大神指点,帮助我解决遇到的问题。
谢谢!