- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
我在 android 中创建了一个音乐播放器服务类,并在 fragment 中有一个播放暂停按钮。我在音乐播放器服务中运行广播流。该服务在后台运行。当应用程序打开服务时,可以通过播放暂停按钮进行控制。但是当应用程序在后台时,如果我关闭服务播放暂停按钮有时不会改变。我该如何解决这个问题
public class RadioStreamFragment extends Fragment {
//Constant
private static final String TAG = "RadioStreamFragment";
//Member variables
private String mTitle = "";
private Bitmap mBitmap;
private boolean mIsPlaying = false;
private int mMaxVolume;
private int mVolume;
private OnSeekBarChangeListener mListener;
private AudioManager mAudioManager;
//View member variables
private TextView mTVTitle;
//private ImageView mImageView;
public static Button mButton;
private SeekBar mSeekBar;
private AutoScrollViewPager pager;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
mAudioManager = (AudioManager) getActivity()
.getSystemService(Context.AUDIO_SERVICE);
}
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState) {
View view = inflater.inflate(R.layout.fragment_radio, null);
//Init view
initView(view);
return view;
}
@Override
public void onResume() {
// TODO Auto-generated method stub
super.onResume();
if (mTitle != null)
setTitleText(mTitle);
if (mBitmap != null)
setImageCover(mBitmap);
//TODO
//Create seekBar state
MainActivity activity = (MainActivity) getActivity();
mMaxVolume = mAudioManager.getStreamMaxVolume(AudioManager.STREAM_MUSIC);
mVolume = mAudioManager.getStreamVolume(AudioManager.STREAM_MUSIC);
mSeekBar.setMax(mMaxVolume);
mSeekBar.setProgress(mVolume);
if (mListener != null)
mSeekBar.setOnSeekBarChangeListener(mListener);
else {
mSeekBar.setOnSeekBarChangeListener(activity.mSeekBarListener);
}
Log.d(TAG, "Volume : " + mVolume + "/" + mMaxVolume);
//Button state
setButtonState(mIsPlaying);
Log.d(TAG, "Resumed");
}
/* Property access functions
*
*/
public void setTitleText(String title) {
mTitle = MainActivity.Songtitle;
mTVTitle.setText(MainActivity.Songtitle);
}
public void setImageCover(Bitmap bitmap) {
mBitmap = bitmap;
// mImageView.setImageBitmap(bitmap);
}
public void setButtonState(boolean isPlaying) {
mIsPlaying = isPlaying;
final int sdk = android.os.Build.VERSION.SDK_INT;
if (isPlaying) {
if(sdk < android.os.Build.VERSION_CODES.JELLY_BEAN) {
mButton.setBackgroundDrawable(getResources()
.getDrawable(R.drawable.btn_pause_radio));
} else {
mButton.setBackground(getResources()
.getDrawable(R.drawable.btn_pause_radio));
}
} else {
if(sdk < android.os.Build.VERSION_CODES.JELLY_BEAN) {
mButton.setBackgroundDrawable(getResources()
.getDrawable(R.drawable.btn_play_radio));
} else {
mButton.setBackground(getResources()
.getDrawable(R.drawable.btn_play_radio));
}
}
}
public void setMaxVolume(int value) {
mMaxVolume = value;
}
public void setVolume(int value) {
mVolume = value;
}
public void setOnSeekBarChangeListener(OnSeekBarChangeListener listener) {
mListener = listener;
}
public int getMaxVolume() {
return mMaxVolume;
}
public int getVolume() {
return mVolume;
}
public SeekBar getVolumeSeekBar() {
return mSeekBar;
}
public String getTitleText() {
return mTitle;
}
/* Internal Functions
*
*/
Button hbir;
Button lb;
private Handler handler;
private void initView(View view) {
final int sdk = android.os.Build.VERSION.SDK_INT;
mTVTitle = (TextView) view.findViewById(R.id.tv_title);
// mImageView = (ImageView) view.findViewById(R.id.im_radio_image);
mButton = (Button) view.findViewById(R.id.btn_play);
mButton.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
// TODO Auto-generated method stub
MainActivity.tryagain=false;
doPlay(v);
}
});
hbir = (Button) view.findViewById(R.id.hbit);
hbir.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
// TODO Auto-generated method stub
hbir.setEnabled(false);
lb.setEnabled(false);
mButton.setEnabled(false);
Data.Initialize();
Data.Initializehbit();
MainActivity.tryagain=false;
try{MainActivity.stopRadio();
MainActivity.playRadio();
if(sdk < android.os.Build.VERSION_CODES.JELLY_BEAN) {
mButton.setBackgroundDrawable(getResources()
.getDrawable(R.drawable.btn_pause_radio));
} else {
mButton.setBackground(getResources()
.getDrawable(R.drawable.btn_pause_radio));
}
as=false;
}
catch(Exception e){}
hbir.setEnabled(false);
lb.setEnabled(true);
mButton.setEnabled(true);
}
});
lb = (Button) view.findViewById(R.id.lbit);
lb.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
// TODO Auto-generated method stub
hbir.setEnabled(false);
lb.setEnabled(false);
mButton.setEnabled(false);
Data.Initialize();
final int sdk = android.os.Build.VERSION.SDK_INT;
MainActivity.tryagain=false;
try{MainActivity.stopRadio();
MainActivity.playRadio();
if(sdk < android.os.Build.VERSION_CODES.JELLY_BEAN) {
mButton.setBackgroundDrawable(getResources()
.getDrawable(R.drawable.btn_pause_radio));
} else {
mButton.setBackground(getResources()
.getDrawable(R.drawable.btn_pause_radio));
}
as=false;
}
catch(Exception e){}
hbir.setEnabled(true);
lb.setEnabled(false);
mButton.setEnabled(true);
}
});
mSeekBar = (SeekBar) view.findViewById(R.id.sb_volume);
mTVTitle.setText(MainActivity.Songtitle);
/* String[] imageUrls = new String[Imageurl.newyearsvalues.size()];
for(int i=0; i<Imageurl.newyearsvalues.size(); i++)
{
ModelWPCategory modelWPCategory = new ModelWPCategory();
modelWPCategory = wpCategories.get(i);
// categoryNames[i]= Imageurl.newyearsvalues.get(i);
Log.d("kd", ""+Imageurl.newyearsvalues.get(i));
imageUrls[i] = Imageurl.newyearsvalues.get(i);;
}
//int pagerPosition = bundle.getInt(Extra.IMAGE_POSITION, 0);
ImagePagerAdapter im = new ImagePagerAdapter(imageUrls);
pager = (AutoScrollViewPager) view.findViewById(R.id.pagera);
pager.setAdapter(im);
pager.setCurrentItem(0);
pager.startAutoScroll();
pager.setScrollbarFadingEnabled(true);
Animation anim = AnimationUtils.loadAnimation(MainActivity.c, R.anim.fade_out);
anim.setRepeatMode(Animation.REVERSE);
pager.startAnimation(anim);
anim.start();*/
/* handler = new Handler();
handler.postDelayed(new Runnable() {
public void run() {
int i = pager.getCurrentItem();
if(i>=(pager.getAdapter().getCount()-1))
i=0;
else
i++;
pager.setCurrentItem(i);
pager.getAdapter().notifyDataSetChanged();
}
},5000);*/
// mTVTitle.setText("Radio Station");
}
public static boolean as=false;
public void doPlay(View view) {
final int sdk = android.os.Build.VERSION.SDK_INT;
if (!mIsPlaying && view.getId() == R.id.btn_play) {
MainActivity.playRadio();
if(as){
try{
if(sdk < android.os.Build.VERSION_CODES.JELLY_BEAN) {
view.setBackgroundDrawable(getResources()
.getDrawable(R.drawable.btn_play_radio));
} else {
view.setBackground(getResources()
.getDrawable(R.drawable.btn_play_radio));
}
}catch(Exception e)
{}
as=false;
}
else
{as=true;
try{
if(sdk < android.os.Build.VERSION_CODES.JELLY_BEAN) {
view.setBackgroundDrawable(getResources()
.getDrawable(R.drawable.btn_pause_radio));
} else {
view.setBackground(getResources()
.getDrawable(R.drawable.btn_pause_radio));
}
}catch(Exception e)
{}
}
} else {
MainActivity.stopRadio();
if(as){
try{
if(sdk < android.os.Build.VERSION_CODES.JELLY_BEAN) {
view.setBackgroundDrawable(getResources()
.getDrawable(R.drawable.btn_play_radio));
} else {
view.setBackground(getResources()
.getDrawable(R.drawable.btn_play_radio));
}
}catch(Exception e)
{}
as=false;
}
else
{as=true;
try{
if(sdk < android.os.Build.VERSION_CODES.JELLY_BEAN) {
view.setBackgroundDrawable(getResources()
.getDrawable(R.drawable.btn_pause_radio));
} else {
view.setBackground(getResources()
.getDrawable(R.drawable.btn_pause_radio));
}
}catch(Exception e)
{}
}
}
}
}
音乐播放器服务类
import java.io.IOException;
import com.spoledge.aacdecoder.MultiPlayer;
import com.spoledge.aacdecoder.PlayerCallback;
import android.app.Service;
import android.content.Context;
import android.content.Intent;
import android.media.AudioManager;
import android.media.AudioTrack;
import android.media.MediaPlayer;
import android.net.ConnectivityManager;
import android.net.NetworkInfo;
import android.os.Binder;
import android.os.Bundle;
import android.os.Handler;
import android.os.IBinder;
import android.util.Log;
import android.widget.Toast;
public class MusicPlayerService extends Service {
//Constant
public static final String NOTIFICATION = "com.vg.intent.notification.musicplayer";
public static final String STATUS = "STATUS";
public static final String STATUS_PLAYING = "Playing";
public static final String STATUS_STOPPED = "Stopped";
public static final String STATUS_BUFFERING = "Buffering";
public static final String STATUS_SERVICE_STARTED = "ServiceStarted";
public static final String PLAY_THIS_ONE = "PlayThisOne";
//Member variables
private static final String TAG = "MusicPlayerSevices";
private StreamBinder mBinder;
private MediaPlayer mMediaPlayer;
private MultiPlayer mPlayer;
private PlayerCallback mPlayerCallback;
private Handler mHandler;
private boolean mIsMP3Pause = false;
//Radio state variables
private String mRadioTitle;
private boolean mIsPlaying = false;
/* Service Lifecycle Event Handler
* (non-Javadoc)
* @see android.app.Service#onCreate()
*/
@Override
public void onCreate() {
mBinder = new StreamBinder();
initMusicPlayer();
super.onCreate();
sendNotification(STATUS, STATUS_SERVICE_STARTED);
mHandler = new Handler();
Log.d(TAG, "onCreate complete");
}
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
if (mBinder == null)
mBinder = new StreamBinder();
if (mPlayer == null)
initMusicPlayer();
if (mMediaPlayer == null)
mMediaPlayer = new MediaPlayer();
handlingRequest(intent);
Log.d(TAG, "onStartCommand complete");
return Service.START_NOT_STICKY; //START_NOT_STICKY still work
}
@Override
public IBinder onBind(Intent intent) {
return mBinder;
}
/* MusicPlayerSevice functions
*
*/
public void playRadio(final String url) {
if (mIsMP3Pause) {
mMediaPlayer.start();
mIsMP3Pause = false;
} else {
//TODO
Thread thread = new Thread(new Runnable() {
@Override
public void run() {
ConnectivityManager connectivityManager = (ConnectivityManager) getSystemService(Context.CONNECTIVITY_SERVICE);
final NetworkInfo networkInfo = connectivityManager.getActiveNetworkInfo();
// Connectivity issue, we quit
if (networkInfo == null || networkInfo.getState() != NetworkInfo.State.CONNECTED) {
return;
}
String newUrl = "";
if (url.contains(".m3u")) {
newUrl = ParserM3UToURL.parse(url);
} else {
newUrl = url;
}
final String finalUrl = newUrl;
mHandler.post(new Runnable() {
@Override
public void run() {
mIsPlaying = true;
if (finalUrl.endsWith(".mp3")) {
//TODO
//Create media player to play instead
Log.d(TAG, "Start media player");
mPlayer.stop();
try {
mMediaPlayer.setDataSource(finalUrl);
mMediaPlayer.prepareAsync();
} catch (IllegalArgumentException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (SecurityException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (IllegalStateException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
mMediaPlayer.start();
mIsMP3Pause = false;
} else {
Log.d(TAG, "Start multi player");
mPlayer.playAsync(finalUrl);
}
}
});
}
});
thread.start();
}
}
public void stopRadio() {
mIsPlaying = false;
if (mMediaPlayer.isPlaying()) {
Log.d(TAG, "Stop media player");
mMediaPlayer.stop();
} else {
Log.d(TAG, "Stop multi player");
mPlayer.stop();
}
}
public void pauseRadio() {
mIsPlaying = false;
if (mMediaPlayer.isPlaying()) {
Log.d(TAG, "Pause media player");
mIsMP3Pause = true;
mMediaPlayer.pause();
} else {
Log.d(TAG, "Stop multi player");
mPlayer.stop();
}
}
public String getRadioTitle() {
return mRadioTitle;
}
public boolean isPlaying() {
return mIsPlaying;
}
//Internal function
private void initMusicPlayer() {
if (mPlayer == null) {
mPlayerCallback = new PlayerCallback() {
@Override
public void playerStopped(int perf) {
sendNotification(STATUS, STATUS_STOPPED);
}
@Override
public void playerStarted() {
sendNotification(STATUS, STATUS_PLAYING);
}
@Override
public void playerPCMFeedBuffer(boolean isPlaying, int bufSizeMs, int bufCapacityMs) {
if (!isPlaying) {
sendNotification(STATUS, STATUS_BUFFERING);
}
}
@Override
public void playerMetadata(String key, String value) {
if (key != null && key.equals("StreamTitle")) {
mRadioTitle = Utils.stripHtml(value);
sendNotification(STATUS, mRadioTitle);
}
}
@Override
public void playerException(Throwable throwable) {
final Throwable finalThrow = throwable;
mHandler.post(new Runnable() {
@Override
public void run() {
stopRadio();
Toast.makeText(getApplicationContext(), finalThrow.getMessage()
, Toast.LENGTH_LONG).show();
sendNotification(STATUS, STATUS_STOPPED);
}
});
}
@Override
public void playerAudioTrackCreated(AudioTrack arg0) {}
};
//Workaround
try {
java.net.URL.setURLStreamHandlerFactory( new java.net.URLStreamHandlerFactory(){
public java.net.URLStreamHandler createURLStreamHandler( String protocol ) {
Log.d( TAG, "Asking for stream handler for protocol: '" + protocol + "'" );
if ("icy".equals( protocol )) return new com.spoledge.aacdecoder.IcyURLStreamHandler();
return null;
}
});
}
catch (Throwable t) {
Log.w( TAG, "Cannot set the ICY URLStreamHandler - maybe already set ? - " + t );
}
mPlayer = new MultiPlayer(mPlayerCallback);
}
}
private void handlingRequest(Intent intent) {
Bundle bundle = intent.getExtras();
if (bundle != null) {
String url = bundle.getString(PLAY_THIS_ONE);
if (url != null) {
playRadio(url);
Log.d(TAG, "Receive playing request : " + url);
}
else {
stopRadio();
Log.d(TAG, "Receive stop request");
}
}
}
private void sendNotification(String key, String value) {
Intent intent = new Intent(NOTIFICATION);
intent.putExtra(key, value);
sendBroadcast(intent);
}
// Nested class
public class StreamBinder extends Binder {
public MusicPlayerService getService() {
return MusicPlayerService.this;
}
}
}
最佳答案
从 fragment/Activity 到服务的通信就像将数据从 Activity 传递到 Activity 一样。
将您的数据(可能是一个 int,表示暂停您的应用程序)放入一个 Intent 中,并使用该 Intent 启动服务。 onStartCommand 将触发,您的服务从 Intent 中提取数据 int 意识到这是一个暂停命令并暂停。
从服务到应用程序的通信很棘手。方法太多,无法在一个答案中涵盖。
关于android - 控制音乐播放器服务 fragment 中的播放暂停按钮,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/29127973/
我最近在/ drawable中添加了一些.gifs,以便可以将它们与按钮一起使用。这个工作正常(没有错误)。现在,当我重建/运行我的应用程序时,出现以下错误: Error: Gradle: Execu
Android 中有返回内部存储数据路径的方法吗? 我有 2 部 Android 智能手机(Samsung s2 和 s7 edge),我在其中安装了一个应用程序。我想使用位于这条路径中的 sqlit
这个问题在这里已经有了答案: What's the difference between "?android:" and "@android:" in an android layout xml f
我只想知道 android 开发手机、android 普通手机和 android root 手机之间的实际区别。 我们不能从实体店或除 android marketplace 以外的其他地方购买开发手
自Gradle更新以来,我正在努力使这个项目达到标准。这是一个团队项目,它使用的是android-apt插件。我已经进行了必要的语法更改(编译->实现和apt->注释处理器),但是编译器仍在告诉我存在
我是android和kotlin的新手,所以请原谅要解决的一个非常简单的问题! 我已经使用导航体系结构组件创建了一个基本应用程序,使用了底部的导航栏和三个导航选项。每个导航选项都指向一个专用片段,该片
我目前正在使用 Facebook official SDK for Android . 我现在正在使用高级示例应用程序,但我不知道如何让它获取应用程序墙/流/状态而不是登录的用户。 这可能吗?在那种情
我在下载文件时遇到问题, 我可以在模拟器中下载文件,但无法在手机上使用。我已经定义了上网和写入 SD 卡的权限。 我在服务器上有一个 doc 文件,如果用户单击下载。它下载文件。这在模拟器中工作正常但
这个问题在这里已经有了答案: What is the difference between gravity and layout_gravity in Android? (22 个答案) 关闭 9
任何人都可以告诉我什么是 android 缓存和应用程序缓存,因为当我们谈论缓存清理应用程序时,它的作用是,缓存清理概念是清理应用程序缓存还是像内存管理一样主存储、RAM、缓存是不同的并且据我所知,缓
假设应用程序 Foo 和 Eggs 在同一台 Android 设备上。任一应用程序都可以获取设备上所有应用程序的列表。一个应用程序是否有可能知道另一个应用程序是否已经运行以及运行了多长时间? 最佳答案
我有点困惑,我只看到了从 android 到 pc 或者从 android 到 pc 的例子。我需要制作一个从两部手机 (android) 连接的 android 应用程序进行视频聊天。我在想,我知道
用于使用 Android 以编程方式锁定屏幕。我从 Stackoverflow 之前关于此的问题中得到了一些好主意,并且我做得很好,但是当我运行该代码时,没有异常和错误。而且,屏幕没有锁定。请在这段代
文档说: android:layout_alignParentStart If true, makes the start edge of this view match the start edge
我不知道这两个属性和高度之间的区别。 以一个TextView为例,如果我将它的layout_width设置为wrap_content,并将它的width设置为50 dip,会发生什么情况? 最佳答案
这两个属性有什么关系?如果我有 android:noHistory="true",那么有 android:finishOnTaskLaunch="true" 有什么意义吗? 最佳答案 假设您的应用中有
我是新手,正在尝试理解以下 XML 代码: 查看 developer.android.com 上的文档,它说“starStyle”是 R.attr 中的常量, public static final
在下面的代码中,为什么当我设置时单选按钮的外观会发生变化 android:layout_width="fill_parent" 和 android:width="fill_parent" 我说的是
很难说出这里要问什么。这个问题模棱两可、含糊不清、不完整、过于宽泛或夸夸其谈,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开,visit the help center . 关闭 9
假设我有一个函数 fun myFunction(name:String, email:String){},当我调用这个函数时 myFunction('Ali', 'ali@test.com ') 如何
我是一名优秀的程序员,十分优秀!