本篇使用Activity来实现一个简单的音乐播放器,提供的功能是在播放音乐的同时可以打开其他的软件,在来电是暂停播放,这里使用到的类有:
android.os.Environment 读取系统存储目录的文件
android.media.MediaPlayer 系统媒体播放器
android.telephony.TelephonyManager 电话管理器
android.telephony.PhoneStateListener 电话状态监听器
我们知道如果不做任何处理,在Activity播放音乐的时候,去打开了另外一个Activity,音乐播放器会仍然播放,因为它没有被摧毁,但是当我们点击模拟器的返回键后,它就会回调onpause()--->onStop()-->onDestroy()方法,此时Activity已经被销毁,我们可以在onPause()方法中来保存进度,然后再onResume()方法中来恢复播放进度。
本实例实现的是播放音乐的同时可以打开其他的软件,当有来电就暂停播放,当电话挂断了就继续播放,这里需要用到的一个权限来监听电话的状态:<uses-permission android:name="android.permission.READ_PHONE_STATE"/>
播放音乐的Activity界面:
当我们点击打开一个activity后,音乐播放器并没有停止(这个不用做任何处理)。
以下是源代码:
main.xml:
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="fill_parent"
android:layout_height="fill_parent"
android:orientation="vertical" >
<TextView
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:text="音乐播放器" />
<EditText
android:id="@+id/filename"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:text="pugongying.mp3"/>
<LinearLayout
android:layout_width="fill_parent"
android:layout_height="fill_parent"
android:orientation="horizontal" >
<Button
android:id="@+id/playButton"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@string/playButton"
android:onClick="mediaplay"/>
<Button
android:id="@+id/pauseButton"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@string/pauseButton"
android:onClick="mediaplay"/>
<Button
android:id="@+id/resetButton"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@string/resetButton"
android:onClick="mediaplay"/>
<Button
android:id="@+id/stopButton"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@string/stopButton"
android:onClick="mediaplay"/>
<Button
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="打开一个activity"
android:onClick="openActivity"/>
</LinearLayout>
</LinearLayout>
values/String.xml
<?xml version="1.0" encoding="utf-8"?>
<resources>
<string name="hello">Hello World, MainActivity!</string>
<string name="app_name">Android_Music</string>
<string name="playButton">播放</string>
<string name="pauseButton">暂停</string>
<string name="resetButton">重播</string>
<string name="stopButton">停止</string>
<string name="continues">继续</string>
</resources>
MainActivity.java
package com.music.activity;
import java.io.File;
import java.io.IOException;
import android.app.Activity;
import android.content.Context;
import android.content.Intent;
import android.media.MediaPlayer;
import android.media.MediaPlayer.OnPreparedListener;
import android.os.Bundle;
import android.os.Environment;
import android.telephony.PhoneStateListener;
import android.telephony.TelephonyManager;
import android.util.Log;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;
import android.widget.Toast;
public class MainActivity extends Activity {
private static String TAG = "MainActivity";
private EditText nameText; //mp3文件名称
private String path; //文件路径
private MediaPlayer mediaPlayer;
private boolean pause; //暂停标志
private int position; //播放进度
@Override
public void onCreate(Bundle savedInstanceState) {
Log.i(TAG, "onCreate()");
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
mediaPlayer = new MediaPlayer();
nameText = (EditText)findViewById(R.id.filename);
TelephonyManager telephonyManager = (TelephonyManager)getSystemService(Context.TELEPHONY_SERVICE);
telephonyManager.listen(new MyPhoneListener(), PhoneStateListener.LISTEN_CALL_STATE);
}
@Override
protected void onDestroy() {
Log.i(TAG, "onDestroy()");
mediaPlayer.release();
mediaPlayer = null;
super.onDestroy();
}
//电话状态监听器
private final class MyPhoneListener extends PhoneStateListener{
@Override
public void onCallStateChanged(int state, String incomingNumber) {
switch (state) {
case TelephonyManager.CALL_STATE_RINGING:
if(mediaPlayer.isPlaying()){
position = mediaPlayer.getCurrentPosition();
mediaPlayer.stop();
}
break;
case TelephonyManager.CALL_STATE_IDLE:
if(position>0 && path!=null){
play();
mediaPlayer.seekTo(position);
position = 0;
}
default:
break;
}
}
}
public void mediaplay(View v){
switch (v.getId()) {
case R.id.playButton:
String fileName = nameText.getText().toString();
File audio = new File(Environment.getExternalStorageDirectory(),fileName);
if(audio.exists()){
path = audio.getAbsolutePath();
play();
}else{
Toast.makeText(getApplicationContext(), fileName+"不存在", 1).show();
}
break;
case R.id.pauseButton:
if(mediaPlayer.isPlaying()){
mediaPlayer.pause();
pause = true;
((Button)v).setText(R.string.continues);
}else{
if(pause){
mediaPlayer.start(); //继续播放
pause = false;
((Button)v).setText(R.string.pauseButton);
}
}
break;
case R.id.resetButton:
if(mediaPlayer.isPlaying()){
mediaPlayer.seekTo(0); //播发进度置0
}else{
if(path!=null){
play();
}
}
break;
case R.id.stopButton:
if(mediaPlayer.isPlaying())
mediaPlayer.stop();
break;
}
}
//播放媒体文件
private void play() {
try {
mediaPlayer.reset(); //重置各项参数
mediaPlayer.setDataSource(path);
mediaPlayer.prepare(); //进行数据缓冲
mediaPlayer.setOnPreparedListener(new PreparedListener());
} catch (IllegalArgumentException e) {
e.printStackTrace();
} catch (IllegalStateException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
}
//监听缓冲完毕事件
private final class PreparedListener implements OnPreparedListener{
//缓冲完毕回调该方法
@Override
public void onPrepared(MediaPlayer mp) {
mediaPlayer.start();; //开始播放
}
}
public void openActivity(View v){
Intent intent = new Intent(MainActivity.this,OtherActivity.class);
startActivity(intent);
}
}
AndroidManifest.xml
<activity android:name=".OtherActivity"/>
<uses-permission android:name="android.permission.READ_PHONE_STATE"/>
分享到:
相关推荐
am命令启动Acitivity流程图
记录归纳了项目里的主要技术点,不同于以往的Activity登录界面,本demo使用了DialogFragment来实现更炫酷好看的效果,并且实现了其与宿主Acitivity的通信,便于完成进一步的网络请求等操作。
AndroidGracePlayer工程:描述:一个音乐播放器具有功能:播放/暂停、上/下一首、后台播放、退出。切换主题。存在的问题:点击Home切换到后台时,进度条时间不会更改。在不同分辨率的手机上,字体大小固定。在...
Android Acitivity的测试代码
实现了日夜模式的切换.(不重启 Acitivity ) 解决了因为快速点击 View 导致的多次响应点击事件. 内部实现了 Android 5.0 的CircularReveal效果. 优点: 布局中直接使用 Android 默认的控件就可以.在解析以后会根据...
activity and Intent 简介 Active/Running 一个新 Activity 启动入栈后,它显示在屏幕最前端,处理是处于栈的最顶端(Activity栈顶),此时它处于可见并可和用户交互的激活状态,叫做活动状态或者运行状态(active or...
该文档主要记录了Activiyt组件启动过程分析,内容参考了Gityuan和老罗的博文。
入门 欢迎来到VS Code Java世界。 这是一条指南,可帮助您开始在Visual Studio Code中编写Java代码。 资料夹结构 默认情况下,工作区包含两个文件夹,其中: src :用于维护源的文件夹 lib :用于维护依赖关系的...
本次项目主要包含了注册、登录和好友列表三个界面以及之间相互跳转。其中好友列表界面设计的很详细,有好友头像和消息内容,登录界面设计的非常好看。 打开应用,进入登录界面,用户可以点击注册按钮进入注册界面,... ...
这个例子讲的是一个activity对应多个acitivity的传值方法
在开发Android app的过程中,遇到这样一个需求:app中启动一个Service,该Service在独立进程中运行,与服务器保持长连接,将服务器推送过来的消息在通知栏中显示,并设置点击动作,点击后跳转到app中对应的Activity...
andriod activity入门级介绍资料,对其生命周期介绍的很好
本文档是自己一边实际操作一边记录完成,每一关键步骤都有截图与详细说明,本是自己的记录也是拿出来与大家分享一下,希望帮助更多的人,因为付出的很多心血,所以没有0积分分享,见谅。方法1的特点是配置简单,缺点...
实现了日夜模式的切换.(不重启 Acitivity ) 解决了因为快速点击 View 导致的多次响应点击事件. 内部实现了 Android 5.0 的CircularReveal效果. 优点: 布局中直接使用 Android 默认的控件就可以.在解析以后会根据控件...
android自定义注解实现findview和click,同时支持Acitivity和Fragment
代码介绍服务的启动两种启动方式,并实现服务于Acitivity 之间的数据交互。
辅助功能,界面左上角显示你当前所在的应用的activity名称,dialog名称,fragment名称等等,可以实时的帮助你调试一个陌生的应用,而不用琐碎的去一个个查看类了。十分方便。
activiti 官网6.0版本-谷歌翻译版
DecoupleMVP###自己写的一个mvp模式,在v和p中间增加了代理层,让两者解藕,并且能过控制view和presenter生命周期的一致性###网络请求####使用了retrofit2 + okhttp+ rxjava2 实现了网络请求,添加了如超时,添加...
用于描述注解的使用范围,可能的ElementType参数如下: CONSTRUCTOR:用于描述构造器 FIELD:用于描述域 LOCAL_VARIABLE:用于描述局部变量 METHOD:用于描述方法 PACKAGE:用于描述包 PARAMETER:用于描述参数 TYPE:用于...