Android Studio音乐播放器

一、本项目完成的功能
1.实现音乐播放器的基本功能:播放音乐,暂停音乐,停止音乐,切换到上一首或者下一首音乐
2.实现一些进阶功能:显示专辑封面、歌曲名称、歌手姓名和进度条
3.调整不同控件的位置、大小,使得整个软件界面看起来大方美观

二、代码部分
1.基本代码
在MainActivity中,初始化一些必要的值,例如音乐播放状态和歌曲名称列表。

public class MainActivity extends Activity implements OnClickListener {
    // 获取界面中显示歌曲标题、作者文本框
    TextView title, author;
    // 播放/暂停、停止按钮
    ImageButton play, stop;
    // 上一首,下一首按钮
    ImageButton pre, next;
    // 获取封面
    ImageView cover;
    ActivityReceiver activityReceiver;

    public static SeekBar audioSeekBar = null;
    public static final String CTL_ACTION =
            "org.ly.action.CTL_ACTION";
    public static final String UPDATE_ACTION =
            "org.ly.action.UPDATE_ACTION";
    // 定义音乐的播放状态,0x11代表没有播放;0x12代表正在播放;0x13代表暂停
    int status = 0x11;
    String[] titleStrs = new String[] { "Ride", "Intentions", "thank u, next" };
    String[] authorStrs = new String[] { "Lana Del Rey", "Justin Bieber", "Ariana Grande" };
    Integer[] covers = new Integer[] { R.drawable.lana, R.drawable.bieber, R.drawable.ariana};

在onCreate函数中,获取页面中各种控件,添加监听器。

@Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        // 获取程序界面界面中的两个按钮
        play = (ImageButton) this.findViewById(R.id.play);
        stop = (ImageButton) this.findViewById(R.id.stop);
        title = (TextView) findViewById(R.id.title);
        author = (TextView) findViewById(R.id.author);
        cover = findViewById(R.id.cover);
        pre = this.findViewById(R.id.previous);
        next = this.findViewById(R.id.next);
        audioSeekBar = (SeekBar) findViewById(R.id.seekBar);

        // 为两个按钮的单击事件添加监听器
        play.setOnClickListener(this);
        stop.setOnClickListener(this);
        // 为上一首、下一首的单击事件添加监听器
        pre.setOnClickListener(this);
        next.setOnClickListener(this);

        activityReceiver = new ActivityReceiver();
        // 创建IntentFilter
        IntentFilter filter = new IntentFilter();
        // 指定BroadcastReceiver监听的Action
        filter.addAction(UPDATE_ACTION);
        // 注册BroadcastReceiver
        registerReceiver(activityReceiver, filter);
        //播放进度监听
        audioSeekBar.setOnSeekBarChangeListener(new SeekBarChangeEvent());
        //保留进度条
        if(MusicService.mPlayer !=null){
            //设置进度条最大值
            MainActivity.audioSeekBar.setMax(MusicService.mPlayer.getDuration());
            audioSeekBar.setProgress(MusicService.mPlayer.getCurrentPosition());
        }
        Intent intent = new Intent(this, MusicService.class);
        // 启动后台Service
        startService(intent);
    }

在onClick函数中,对于不同按钮的点击实现不同操作。

@Override
    public void onClick(View view) {
        // 创建Intent
        Intent intent = new Intent("org.ly.action.CTL_ACTION");
        switch (view.getId())
        {
            // 按下播放/暂停按钮
            case R.id.play:
                intent.putExtra("control", 1);
                break;
            // 按下停止按钮
            case R.id.stop:
                intent.putExtra("control", 2);
                break;
            case R.id.previous:
                intent.putExtra("control",3);
            case R.id.next:
                intent.putExtra("control",4);
        }
        // 发送广播,将被Service组件中的BroadcastReceiver接收到
        sendBroadcast(intent);
    }

在MusicService中,也是先初始化一些必要的值。

public class MusicService extends Service implements Runnable {
    MyReceiver serviceReceiver;
    AssetManager am;
    String[] musics = new String[] { "Ride.mp3","Intentions.mp3","thank u, next.mp3" };
    static MediaPlayer mPlayer;
    // 当前的状态,0x11代表没有播放;0x12代表正在播放;0x13代表暂停
    int status = 0x11;
    // 记录当前正在播放的音乐
    int current = 0;
    @Override
    public IBinder onBind(Intent intent)
    {
        return null;
    }

然后在prepareAndPlay函数中实现播放音乐的代码。

private void prepareAndPlay(String music) {
        try {
            // 打开指定音乐文件
            AssetFileDescriptor afd = am.openFd(music);
            mPlayer.reset();
            // 使用MediaPlayer加载指定的声音文件。
            mPlayer.setDataSource(afd.getFileDescriptor(),
                    afd.getStartOffset(), afd.getLength());
            // 准备声音
            mPlayer.prepare();
            // 播放
            mPlayer.start();
            MainActivity.audioSeekBar.setMax(this.mPlayer.getDuration());
            new Thread(this).start();
        }
        catch (IOException e) {
            e.printStackTrace();
        }
    }

2.重要代码部分
本次实验的难点在于在两个类MainActivity和MusicService中分别实现两个广播。MainActivity只负责与页面有关的功能,比如页面显示,监测页面点击等。MusicService只负责与音乐播放有关的功能,比如音乐的暂停和切换等。这两部分功能在两个函数中,但又需要互通消息,因此采用广播机制实现消息的发送和接受。

第一个需求是MusicService在切换歌曲的时候,需要向MainActivity发送消息,告诉MainActivity需要修改页面的文本框信息了。
具体步骤是在MusicService中的onCreate函数中,编写歌曲播放代码,并实现一首音乐播放完毕就向MainActivity发送一个intent。

@Override
    public void onCreate() {
        super.onCreate();
        am = getAssets();
        // 创建BroadcastReceiver
        serviceReceiver = new MyReceiver();
        // 创建IntentFilter
        IntentFilter filter = new IntentFilter();
        filter.addAction(MainActivity.CTL_ACTION);
        registerReceiver(serviceReceiver, filter);
        // 创建MediaPlayer
        mPlayer = new MediaPlayer();
        // 为MediaPlayer播放完成事件绑定监听器
        mPlayer.setOnCompletionListener(new OnCompletionListener(){
            @Override
            public void onCompletion(MediaPlayer mp) {
                current++;
                MainActivity.audioSeekBar.setMax(0);
                if (current >= 3) {
                    current = 0;
                }
                //发送广播通知Activity更改文本框
                Intent sendIntent = new Intent(MainActivity.UPDATE_ACTION);
                sendIntent.putExtra("current", current);
                // 发送广播,将被Activity组件中的BroadcastReceiver接收到
                sendBroadcast(sendIntent);
                // 准备并播放音乐
                prepareAndPlay(musics[current]);
            }
        });
    }

MainActivity需要接收广播,那么就在MainActivity中定义一个ActivityReceiver子类,用来获取MusicService正在播放的歌曲信息。具体步骤还有定义过滤器,注册Receiver,启动intent服务。

public class ActivityReceiver extends BroadcastReceiver {
        @Override
        public void onReceive(Context context, Intent intent) {
            // 获取Intent中的update消息,update代表播放状态
            int update = intent.getIntExtra("update", -1);
            // 获取Intent中的current消息,current代表当前正在播放的歌曲
            int current = intent.getIntExtra("current", -1);
            if (current >= 0) {
                title.setText(titleStrs[current]);
                author.setText(authorStrs[current]);
                cover.setImageResource(covers[current]);
            }
            switch (update) {
                case 0x11:
                    play.setImageResource(R.drawable.play);
                    status = 0x11;
                    break;
                // 控制系统进入播放状态
                case 0x12:
                    // 播放状态下设置使用暂停图标
                    play.setImageResource(R.drawable.pause);
                    // 设置当前状态
                    status = 0x12;
                    break;
                // 控制系统进入暂停状态
                case 0x13:
                    // 暂停状态下设置使用播放图标
                    play.setImageResource(R.drawable.play);
                    // 设置当前状态
                    status = 0x13;
                    break;
            }
        }
    }

第二个需求是MainActivity监测到的用户对页面按钮的点击请求,自己不能实现,需要发送给MusicService让其实现。第二个广播在Activity中的onClick函数中定义intent,对接受到的不同的按钮控制信息向MusicService发送控制信息。然后在MusicService中定义Receiver。

这里需要注意的是,要接收广播,在Activity中用ActivityReceiver类,在Service用BroadcastReceiver类。

三、效果展示

android stdio 放音乐 android studio设计音乐播放器_音乐播放

android stdio 放音乐 android studio设计音乐播放器_ide_02

四、代码实现
最后附上项目具体源码地址:https://gitee.com/CieloSs/mobile-development-music-box