Android應用開發 MP3音樂播放器程式碼實現 二
Android應用開發--MP3音樂播放器程式碼實現(二)
2013年5月25日 簡、美音樂播放器開發
小巫在這裡羅列這個播放器已經實現的功能:
1. 自動顯示音樂列表
2. 點選列表播放音樂
3. 長按列表彈出對話方塊
4. 暫停音樂
5. 上一首音樂
6. 下一首音樂
7. 自動播放下一首歌曲
8. 單曲迴圈
9. 全部迴圈
10. 隨機播放
以上所有功能將會分為兩篇博文來講解,首先是主介面的,接著是播放介面的。在這裡要說明一點,以上功能是小巫自己一點一點除錯才實現的,並不能完全排除考慮不周的地方,原本這個軟體實現起來並不太難,但確實要考慮到很多細節的地方,播放狀態的切換和控制就是一塊,也花了我不少實現,之前還很苦惱實現自己想要的效果,但後來還是經過思考和除錯把功能實現。所以說,開發是一個需要很耐心的過程,各位童鞋,如果真正喜歡程式設計的話,想要做出一些小作品的話,那就好好掂量自己的耐心吧,好了,廢話不多說,先貼一大段程式碼,後面在慢慢把需要注意的地方說一下。
主介面效果圖:
以上介面的效果怎麼實現的?
很簡單的,就是ListView的資料填充,但要填的的東西就要考慮了,怎麼把資料從SQLite中獲取,小巫封裝了一個工具類,用來獲取與MP3相關的資料。
==>MediaUtils
package com.wwj.sb.utils;import java.util.ArrayList;import java.util.HashMap;import java.util.Iterator;import java.util.List;import android.content.Context;import android.database.Cursor;import android.provider.MediaStore;import com.wwj.sb.domain.Mp3Info;public class MediaUtil { /** * 用於從資料庫中查詢歌曲的資訊,儲存在List當中 * * @return */ public static List<Mp3Info> getMp3Infos(Context context) { Cursor cursor = context.getContentResolver().query( MediaStore.Audio.Media.EXTERNAL_CONTENT_URI, null , null, null, MediaStore.Audio.Media.DEFAULT_SORT_ORDER); List<Mp3Info> mp3Infos = new ArrayList<Mp3Info>(); for (int i = 0; i < cursor.getCount(); i++) { cursor.moveToNext(); Mp3Info mp3Info = new Mp3Info(); long id = cursor.getLong(cursor .getColumnIndex(MediaStore.Audio.Media._ID)); //音樂id String title = cursor.getString((cursor .getColumnIndex(MediaStore.Audio.Media.TITLE))); //音樂標題 String artist = cursor.getString(cursor .getColumnIndex(MediaStore.Audio.Media.ARTIST)); //藝術家 long duration = cursor.getLong(cursor .getColumnIndex(MediaStore.Audio.Media.DURATION)); //時長 long size = cursor.getLong(cursor .getColumnIndex(MediaStore.Audio.Media.SIZE)); //檔案大小 String url = cursor.getString(cursor .getColumnIndex(MediaStore.Audio.Media.DATA)); //檔案路徑 int isMusic = cursor.getInt(cursor .getColumnIndex(MediaStore.Audio.Media.IS_MUSIC)); //是否為音樂 if (isMusic != 0) { //只把音樂新增到集合當中 mp3Info.setId(id); mp3Info.setTitle(title); mp3Info.setArtist(artist); mp3Info.setDuration(duration); mp3Info.setSize(size); mp3Info.setUrl(url); mp3Infos.add(mp3Info); } } return mp3Infos; } /** * 往List集合中新增Map物件資料,每一個Map物件存放一首音樂的所有屬性 * @param mp3Infos * @return */ public static List<HashMap<String, String>> getMusicMaps( List<Mp3Info> mp3Infos) { List<HashMap<String, String>> mp3list = new ArrayList<HashMap<String, String>>(); for (Iterator iterator = mp3Infos.iterator(); iterator.hasNext();) { Mp3Info mp3Info = (Mp3Info) iterator.next(); HashMap<String, String> map = new HashMap<String, String>(); map.put("title", mp3Info.getTitle()); map.put("Artist", mp3Info.getArtist()); map.put("duration", formatTime(mp3Info.getDuration())); map.put("size", String.valueOf(mp3Info.getSize())); map.put("url", mp3Info.getUrl()); mp3list.add(map); } return mp3list; } /** * 格式化時間,將毫秒轉換為分:秒格式 * @param time * @return */ public static String formatTime(long time) { String min = time / (1000 * 60) + ""; String sec = time % (1000 * 60) + ""; if (min.length() < 2) { min = "0" + time / (1000 * 60) + ""; } else { min = time / (1000 * 60) + ""; } if (sec.length() == 4) { sec = "0" + (time % (1000 * 60)) + ""; } else if (sec.length() == 3) { sec = "00" + (time % (1000 * 60)) + ""; } else if (sec.length() == 2) { sec = "000" + (time % (1000 * 60)) + ""; } else if (sec.length() == 1) { sec = "0000" + (time % (1000 * 60)) + ""; } return min + ":" + sec.trim().substring(0, 2); }}
好吧,來重頭戲了,一大段程式碼來襲。
HomeActivity.Java
package com.wwj.sb.activity;import java.util.HashMap;import java.util.List;import android.app.Activity;import android.app.AlertDialog;import android.app.Service;import android.content.BroadcastReceiver;import android.content.Context;import android.content.DialogInterface;import android.content.Intent;import android.content.IntentFilter;import android.graphics.Color;import android.os.Bundle;import android.os.Vibrator;import android.view.ContextMenu;import android.view.ContextMenu.ContextMenuInfo;import android.view.KeyEvent;import android.view.View;import android.view.View.OnClickListener;import android.view.View.OnCreateContextMenuListener;import android.view.ViewGroup.LayoutParams;import android.widget.AdapterView;import android.widget.AdapterView.OnItemClickListener;import android.widget.ArrayAdapter;import android.widget.Button;import android.widget.ImageView;import android.widget.ListView;import android.widget.SimpleAdapter;import android.widget.TextView;import android.widget.Toast;import com.wwj.sb.domain.AppConstant;import com.wwj.sb.domain.Mp3Info;import com.wwj.sb.service.PlayerService;import com.wwj.sb.utils.ConstantUtil;import com.wwj.sb.utils.CustomDialog;import com.wwj.sb.utils.MediaUtil;/** * 2013/5/7 * 簡、美音樂播放器 * @author wwj * */public class HomeActivity extends Activity { private ListView mMusiclist; // 音樂列表 private List<Mp3Info> mp3Infos = null; private SimpleAdapter mAdapter; // 簡單介面卡 private Button previousBtn; // 上一首 private Button repeatBtn; // 重複(單曲迴圈、全部迴圈) private Button playBtn; // 播放(播放、暫停) private Button shuffleBtn; // 隨機播放 private Button nextBtn; // 下一首 private TextView musicTitle;//歌曲標題 private TextView musicDuration; //歌曲時間 private Button musicPlaying; //歌曲專輯 private int repeatState; //迴圈標識 private final int isCurrentRepeat = 1; // 單曲迴圈 private final int isAllRepeat = 2; // 全部迴圈 private final int isNoneRepeat = 3; // 無重複播放 private boolean isFirstTime = true; private boolean isPlaying; // 正在播放 private boolean isPause; // 暫停 private boolean isNoneShuffle = true; // 順序播放 private boolean isShuffle = false; // 隨機播放 private int listPosition = 0; //標識列表位置 private HomeReceiver homeReceiver; //自定義的廣播接收器 //一系列動作 public static final String UPDATE_ACTION = "com.wwj.action.UPDATE_ACTION"; public static final String CTL_ACTION = "com.wwj.action.CTL_ACTION"; public static final String MUSIC_CURRENT = "com.wwj.action.MUSIC_CURRENT"; public static final String MUSIC_DURATION = "com.wwj.action.MUSIC_DURATION"; public static final String REPEAT_ACTION = "com.wwj.action.REPEAT_ACTION"; public static final String SHUFFLE_ACTION = "com.wwj.action.SHUFFLE_ACTION"; private int currentTime; private int duration; @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.home_activity_layout); mMusiclist = (ListView) findViewById(R.id.music_list); mMusiclist.setOnItemClickListener(new MusicListItemClickListener()); mMusiclist.setOnCreateContextMenuListener(new MusicListItemContextMenuListener()); mp3Infos = MediaUtil.getMp3Infos(getApplicationContext()); //獲取歌曲物件集合 setListAdpter(MediaUtil.getMusicMaps(mp3Infos)); //顯示歌曲列表 findViewById(); //找到介面上的每一個控制元件 setViewOnclickListener(); //為一些控制元件設定監聽器 repeatState = isNoneRepeat; // 初始狀態為無重複播放狀態 homeReceiver = new HomeReceiver(); // 建立IntentFilter IntentFilter filter = new IntentFilter(); // 指定BroadcastReceiver監聽的Action filter.addAction(UPDATE_ACTION); filter.addAction(MUSIC_CURRENT); filter.addAction(MUSIC_DURATION); filter.addAction(REPEAT_ACTION); filter.addAction(SHUFFLE_ACTION); // 註冊BroadcastReceiver registerReceiver(homeReceiver, filter); } /** * 從介面上根據id獲取按鈕 */ private void findViewById() { previousBtn = (Button) findViewById(R.id.previous_music); repeatBtn = (Button) findViewById(R.id.repeat_music); playBtn = (Button) findViewById(R.id.play_music); shuffleBtn = (Button) findViewById(R.id.shuffle_music); nextBtn = (Button) findViewById(R.id.next_music); musicTitle = (TextView) findViewById(R.id.music_title); musicDuration = (TextView) findViewById(R.id.music_duration); musicPlaying = (Button) findViewById(R.id.playing); } /** * 給每一個按鈕設定監聽器 */ private void setViewOnclickListener() { ViewOnClickListener viewOnClickListener = new ViewOnClickListener(); previousBtn.setOnClickListener(viewOnClickListener); repeatBtn.setOnClickListener(viewOnClickListener); playBtn.setOnClickListener(viewOnClickListener); shuffleBtn.setOnClickListener(viewOnClickListener); nextBtn.setOnClickListener(viewOnClickListener); musicPlaying.setOnClickListener(viewOnClickListener); } private class ViewOnClickListener implements OnClickListener { Intent intent = new Intent(); @Override public void onClick(View v) { switch (v.getId()) { case R.id.previous_music: // 上一首 playBtn.setBackgroundResource(R.drawable.play_selector); isFirstTime = false; isPlaying = true; isPause = false; previous(); break; case R.id.repeat_music: // 重複播放 if (repeatState == isNoneRepeat) { repeat_one(); shuffleBtn.setClickable(false); repeatState = isCurrentRepeat; } else if (repeatState == isCurrentRepeat) { repeat_all(); shuffleBtn.setClickable(false); repeatState = isAllRepeat; } else if (repeatState == isAllRepeat) { repeat_none(); shuffleBtn.setClickable(true); repeatState = isNoneRepeat; } switch (repeatState) { case isCurrentRepeat: // 單曲迴圈 repeatBtn .setBackgroundResource(R.drawable.repeat_current_selector); Toast.makeText(HomeActivity.this, R.string.repeat_current, Toast.LENGTH_SHORT).show(); break; case isAllRepeat: // 全部迴圈 repeatBtn .setBackgroundResource(R.drawable.repeat_all_selector); Toast.makeText(HomeActivity.this, R.string.repeat_all, Toast.LENGTH_SHORT).show(); break; case isNoneRepeat: // 無重複 repeatBtn .setBackgroundResource(R.drawable.repeat_none_selector); Toast.makeText(HomeActivity.this, R.string.repeat_none, Toast.LENGTH_SHORT).show(); break; } break; case R.id.play_music: // 播放音樂 if(isFirstTime) { play(); isFirstTime = false; isPlaying = true; isPause = false; } else { if (isPlaying) { playBtn.setBackgroundResource(R.drawable.pause_selector); intent.setAction("com.wwj.media.MUSIC_SERVICE"); intent.putExtra("MSG", AppConstant.PlayerMsg.PAUSE_MSG); startService(intent); isPlaying = false; isPause = true; } else if (isPause) { playBtn.setBackgroundResource(R.drawable.play_selector); intent.setAction("com.wwj.media.MUSIC_SERVICE"); intent.putExtra("MSG", AppConstant.PlayerMsg.CONTINUE_MSG); startService(intent); isPause = false; isPlaying = true; } } break; case R.id.shuffle_music: // 隨機播放 if (isNoneShuffle) { shuffleBtn .setBackgroundResource(R.drawable.shuffle_selector); Toast.makeText(HomeActivity.this, R.string.shuffle, Toast.LENGTH_SHORT).show(); isNoneShuffle = false; isShuffle = true; shuffleMusic(); repeatBtn.setClickable(false); } else if (isShuffle) { shuffleBtn .setBackgroundResource(R.drawable.shuffle_none_selector); Toast.makeText(HomeActivity.this, R.string.shuffle_none, Toast.LENGTH_SHORT).show(); isShuffle = false; isNoneShuffle = true; repeatBtn.setClickable(true); } break; case R.id.next_music: // 下一首 playBtn.setBackgroundResource(R.drawable.play_selector); isFirstTime = false; isPlaying = true; isPause = false; next(); break; case R.id.playing: //正在播放 Mp3Info mp3Info = mp3Infos.get(listPosition); Intent intent = new Intent(HomeActivity.this, PlayerActivity.class); intent.putExtra("title", mp3Info.getTitle()); intent.putExtra("url", mp3Info.getUrl()); intent.putExtra("artist", mp3Info.getArtist()); intent.putExtra("listPosition", listPosition); intent.putExtra("currentTime", currentTime); intent.putExtra("duration", duration); intent.putExtra("MSG", AppConstant.PlayerMsg.PLAYING_MSG); startActivity(intent); break; } } } private class MusicListItemClickListener implements OnItemClickListener { /** * 點選列表播放音樂 */ @Override public void onItemClick(AdapterView<?> parent, View view, int position, long id) { listPosition = position; playMusic(listPosition); } } public class MusicListItemContextMenuListener implements OnCreateContextMenuListener { @Override public void onCreateContextMenu(ContextMenu menu, View v, ContextMenuInfo menuInfo) { Vibrator vibrator = (Vibrator) getSystemService(Service.VIBRATOR_SERVICE); vibrator.vibrate(50); //長按振動 musicListItemDialog(); //長按後彈出的對話方塊 } } /** * 填充列表 * * @param mp3Infos */ public void setListAdpter(List<HashMap<String, String>> mp3list) { mAdapter = new SimpleAdapter(this, mp3list, R.layout.music_list_item_layout, new String[] { "title", "Artist", "duration" }, new int[] { R.id.music_title, R.id.music_Artist, R.id.music_duration }); mMusiclist.setAdapter(mAdapter); } /** * 下一首歌曲 */ public void next() { listPosition = listPosition + 1; if(listPosition <= mp3Infos.size() - 1) { Mp3Info mp3Info = mp3Infos.get(listPosition); musicTitle.setText(mp3Info.getTitle()); Intent intent = new Intent(); intent.setAction("com.wwj.media.MUSIC_SERVICE"); intent.putExtra("listPosition", listPosition); intent.putExtra("url", mp3Info.getUrl()); intent.putExtra("MSG", AppConstant.PlayerMsg.NEXT_MSG); startService(intent); } else { Toast.makeText(HomeActivity.this, "沒有下一首了", Toast.LENGTH_SHORT).show(); } } /** * 上一首歌曲 */ public void previous() { listPosition = listPosition - 1; if(listPosition >= 0) { Mp3Info mp3Info = mp3Infos.get(listPosition); musicTitle.setText(mp3Info.getTitle()); Intent intent = new Intent(); intent.setAction("com.wwj.media.MUSIC_SERVICE"); intent.putExtra("listPosition", listPosition); intent.putExtra("url", mp3Info.getUrl()); intent.putExtra("MSG", AppConstant.PlayerMsg.PRIVIOUS_MSG); startService(intent); }else { Toast.makeText(HomeActivity.this, "沒有上一首了", Toast.LENGTH_SHORT).show(); } } public void play() { playBtn.setBackgroundResource(R.drawable.play_selector); Mp3Info mp3Info = mp3Infos.get(listPosition); musicTitle.setText(mp3Info.getTitle()); Intent intent = new Intent(); intent.setAction("com.wwj.media.MUSIC_SERVICE"); intent.putExtra("listPosition", 0); intent.putExtra("url", mp3Info.getUrl()); intent.putExtra("MSG", AppConstant.PlayerMsg.PLAY_MSG); startService(intent); } /** * 單曲迴圈 */ public void repeat_one() { Intent intent = new Intent(CTL_ACTION); intent.putExtra("control", 1); sendBroadcast(intent); } /** * 全部迴圈 */ public void repeat_all() { Intent intent = new Intent(CTL_ACTION); intent.putExtra("control", 2); sendBroadcast(intent); } /** * 順序播放 */ public void repeat_none() { Intent intent = new Intent(CTL_ACTION); intent.putExtra("control", 3); sendBroadcast(intent); } /** * 隨機播放 */ public void shuffleMusic() { Intent intent = new Intent(CTL_ACTION); intent.putExtra("control", 4); sendBroadcast(intent); } public void musicListItemDialog() { String[] menuItems = new String[]{"播放音樂","設為鈴聲","檢視詳情"}; ListView menuList = new ListView(HomeActivity.this); menuList.setCacheColorHint(Color.TRANSPARENT); menuList.setDividerHeight(1); menuList.setAdapter(new ArrayAdapter<String>(HomeActivity.this, R.layout.context_dialog_layout, R.id.dialogText, menuItems)); menuList.setLayoutParams(new LayoutParams(ConstantUtil.getScreen(HomeActivity.this)[0] / 2, LayoutParams.WRAP_CONTENT)); final CustomDialog customDialog = new CustomDialog.Builder(HomeActivity.this).setTitle(R.string.operation).setView(menuList).create(); customDialog.show(); menuList.setOnItemClickListener( new OnItemClickListener() { @Override public void onItemClick(AdapterView<?> parent, View view, int position, long id) { customDialog.cancel(); customDialog.dismiss(); } }); } public void playMusic(int listPosition) { if (mp3Infos != null) { Mp3Info mp3Info = mp3Infos.get(listPosition); musicTitle.setText(mp3Info.getTitle()); Intent intent = new Intent(HomeActivity.this, PlayerActivity.class); intent.putExtra("title", mp3Info.getTitle()); intent.putExtra("url", mp3Info.getUrl()); intent.putExtra("artist", mp3Info.getArtist()); intent.putExtra("listPosition", listPosition); intent.putExtra("currentTime", currentTime); intent.putExtra("repeatState", repeatState); intent.putExtra("shuffleState", isShuffle); intent.putExtra("MSG", AppConstant.PlayerMsg.PLAY_MSG); startActivity(intent); } } @Override protected void onStop() { // TODO Auto-generated method stub super.onStop(); } @Override protected void onDestroy() { // TODO Auto-generated method stub super.onDestroy(); } /** * 按返回鍵彈出對話方塊確定退出 */ @Override public boolean onKeyDown(int keyCode, KeyEvent event) { if (keyCode == KeyEvent.KEYCODE_BACK && event.getAction() == KeyEvent.ACTION_DOWN) { new AlertDialog.Builder(this) .setIcon(R.drawable.ic_launcher) .setTitle("退出") .setMessage("您確定要退出?") .setNegativeButton("取消", null) .setPositiveButton("確定", new DialogInterface.OnClickListener() { @Override public void onClick(DialogInterface dialog, int which) { finish(); Intent intent = new Intent( HomeActivity.this, PlayerService.class); unregisterReceiver(homeReceiver); stopService(intent); // 停止後臺服務 } }).show(); } return super.onKeyDown(keyCode, event); } //自定義的BroadcastReceiver,負責監聽從Service傳回來的廣播 public class HomeReceiver extends BroadcastReceiver { @Override public void onReceive(Context context, Intent intent) { String action = intent.getAction(); if(action.equals(MUSIC_CURRENT)){ //currentTime代表當前播放的時間 currentTime = intent.getIntExtra("currentTime", -1); musicDuration.setText(MediaUtil.formatTime(currentTime)); } else if (action.equals(MUSIC_DURATION)) { duration = intent.getIntExtra("duration", -1); } else if(action.equals(UPDATE_ACTION)) { //獲取Intent中的current訊息,current代表當前正在播放的歌曲 listPosition = intent.getIntExtra("current", -1); if(listPosition >= 0) { musicTitle.setText(mp3Infos.get(listPosition).getTitle()); } }else if(action.equals(REPEAT_ACTION)) { repeatState = intent.getIntExtra("repeatState", -1); switch (repeatState) { case isCurrentRepeat: // 單曲迴圈 repeatBtn .setBackgroundResource(R.drawable.repeat_current_selector); shuffleBtn.setClickable(false); break; case isAllRepeat: // 全部迴圈 repeatBtn .setBackgroundResource(R.drawable.repeat_all_selector); shuffleBtn.setClickable(false); break; case isNoneRepeat: // 無重複 repeatBtn .setBackgroundResource(R.drawable.repeat_none_selector); shuffleBtn.setClickable(true); break; } } else if(action.equals(SHUFFLE_ACTION)) { isShuffle = intent.getBooleanExtra("shuffleState", false); if(isShuffle) { isNoneShuffle = false; shuffleBtn.setBackgroundResource(R.drawable.shuffle_selector); repeatBtn.setClickable(false); } else { isNoneShuffle = true; shuffleBtn.setBackgroundResource(R.drawable.shuffle_none_selector); repeatBtn.setClickable(true); } } } }}
到這裡,要開講啦。
以下是需要注意的幾點:
1. 音樂是通過Service來播放的,Activity通過啟動服務來實現在後臺播放音樂。
2. Activity中自定義了一個廣播接收器,需要進行intent過濾器的定義,動作的新增,註冊廣播接收器:
homeReceiver = new HomeReceiver(); // 建立IntentFilter IntentFilter filter = new IntentFilter(); // 指定BroadcastReceiver監聽的Action filter.addAction(UPDATE_ACTION); filter.addAction(MUSIC_CURRENT); filter.addAction(MUSIC_DURATION); filter.addAction(REPEAT_ACTION); filter.addAction(SHUFFLE_ACTION); // 註冊BroadcastReceiver registerReceiver(homeReceiver, filter);
3. 在廣播接收器類當中對動作進行處理,比如實現時間的更新和標題的更新等。
4. 這裡還要注意按鈕觸發,播放狀態的改變,比如音樂迴圈,有三種狀態:單曲、全部迴圈、順序,每切換一個狀態都要向服務傳送一條廣播,通知它要改變狀態。
5. 點選列表的時候,會跳入到播放介面的Activity中,要注意用intent來傳遞引數,注意每個引數的用途,比如title、url、MSG,就分別代表標題、路徑、播放狀態。
6. 長按列表會彈出自定義對話方塊,也會有短暫的震動效果,自定義對話方塊需要自行實現。這裡我也貼一下實現程式碼吧。
package com.wwj.sb.utils;import android.app.Activity;import android.app.Dialog;import android.content.Context;import android.content.DialogInterface;import android.text.TextUtils;import android.view.LayoutInflater;import android.view.View;import android.view.ViewGroup;import android.view.ViewGroup.LayoutParams;import android.widget.Button;import android.widget.FrameLayout;import android.widget.ImageView;import android.widget.LinearLayout;import android.widget.TextView;import com.wwj.sb.activity.R;/** * 自定義對話方塊類 * * @author wwj * */public class CustomDialog extends Dialog { public CustomDialog(Context context) { super(context); } public CustomDialog(Context context, int theme) { super(context, theme); } public static class Builder { private Context context; private int mIcon = -1; // 提示圖示 private CharSequence mTitle; // 提示標題 private CharSequence mMessage; // 提示內容 private CharSequence mPositiveButtonText; // 確定按鈕文字 private CharSequence mNegativeButtonText; // 取消按鈕文字 private CharSequence mNeutralButtonText; // 中間按鈕文字 private boolean mCancelable = true; // 是否啟用取消鍵 private int mViewSpacingLeft; private int mViewSpacingTop; private int mViewSpacingRight; private int mViewSpacingBottom; private boolean mViewSpacingSpecified = false; // 提示內容View private