热门标签 | HotTags
当前位置:  开发笔记 > Android > 正文

Android开发之第一个小项目--我的MP3播放器之主界面的实现(二)

嗯先上两张图~~这里两张图片,第一张是列表界面,也是主界面,能显示歌曲界面个下面的播放栏,播放栏点击能进入第二个界面,也能够控制歌曲的播放个暂停。第二个界面是播放界面,这个界面

嗯 先上两张图~~
);
qq.setMusicUrl("http://music.baidu.com/search?key=" + songList.get(songListPosition).getTitle());
Platform qqq = ShareSDK.getPlatform(MainActivity.this, QQ.NAME);
qqq.setPlatformActionListener(MainActivity.this);
qqq.share(qq);
break;
case R.id.action_settings:
unregisterReceiver(mainReceiver);
Intent intent = new Intent(MainActivity.this,MusicPlayService.class);
stopService(intent);
isRealDestroy = true;
MainActivity.this.finish();

break;

}
if (!msg.equals("")) {
Toast.makeText(MainActivity.this, msg, Toast.LENGTH_SHORT).show();
}
return true;
}
});
// getSupportActionBar().setDisplayHomeAsUpEnabled(false);左上角那个箭头
values = new ContentValues();
/**
* 这下面的功能是保存当前正在播放的歌曲的信息,因为我写的功能是退出了主界面之后,歌曲在后台播放,
* 但是我一旦点击程序,下面还是能够与显示当前正在播放的歌曲,这个就要保存数据了,所以我先建立了一个数据库,
* 后面换了歌曲的话会把当前播放的歌曲信息在这个数据库中更新,但是,要先这个数据库中有数据才能更新,所以我
* 先往这个数据库中先存入了一个数据,就是列表的第一首歌,好方便后面更新。
*/

int result = DataSupport.count(PlaySong.class);
if (result == 0) {
//当前播放歌曲的实体类
PlaySong playSOng= new PlaySong();
playSong.setId(1);
playSong.setSongPosition(-1);
playSong.setPlaying(false);
playSong.save();
}
/**
* 这是一个加载图片的线程,没一个歌曲都有它的对应图片,这个线程是取出歌曲里面的图片
* ,然后剪切成圆形,给后面使用
*/

if (sOngBitmapThread== null) {
sOngBitmapThread= new SongBitmapThread();
}
//初始化View
createView();
//设置数据
setAdapter();

}

// 通知栏显示当前播放信息
public void showNotification() {
notificatiOnManager= (NotificationManager) getSystemService(NOTIFICATION_SERVICE);
Notification notification = new Notification(R.drawable.like,
"歌曲:"+songList.get(songListPosition).getTitle() + "--" + "歌手:"+songList.get(songListPosition).getArtist(),
System.currentTimeMillis());

notification.icon = R.drawable.bg;//notification通知栏图标

//自定义布局
cOntentView= new RemoteViews(getPackageName(), R.layout.notification_item);
if (songList.get(songListPosition).getBitmap() == null) {

Resources res = getResources();
Bitmap bmp = BitmapFactory.decodeResource(res, R.drawable.morenbgsamall);
contentView.setImageViewBitmap(R.id.im, bmp);
} else {
contentView.setImageViewBitmap(R.id.im, songList.get(songListPosition).getBitmap());
}
/**
* 这下面一大段是通知栏上的按键来控制歌曲的播放,
* 这个不懂的话可以去查找下Notification的用法
* prev 前一首
* play 播放or暂停
* next 下一首
*/

contentView.setTextViewText(R.id.notification_title,songList.get(songListPosition).getTitle());
contentView.setTextViewText(R.id.notification_artist,songList.get(songListPosition).getArtist());
Intent noticifitiOnMsg= new Intent(AppControl.NOTIFICATION_ACTION);
noticifitionMsg.putExtra("INTENT_BUTTONID_TAG", "prev");
PendingIntent pendingIntent = PendingIntent.getBroadcast(this, 1, noticifitionMsg, PendingIntent.FLAG_CANCEL_CURRENT);
contentView.setOnClickPendingIntent(R.id.last, pendingIntent);

noticifitionMsg.putExtra("INTENT_BUTTONID_TAG", "play");
PendingIntent pendingIntent2 = PendingIntent.getBroadcast(this, 2, noticifitionMsg, PendingIntent.FLAG_CANCEL_CURRENT);
contentView.setOnClickPendingIntent(R.id.now, pendingIntent2);

noticifitionMsg.putExtra("INTENT_BUTTONID_TAG", "next");
PendingIntent pendingIntent3 = PendingIntent.getBroadcast(this, 3, noticifitionMsg, PendingIntent.FLAG_CANCEL_CURRENT);
contentView.setOnClickPendingIntent(R.id.future, pendingIntent3);

notification.cOntentView= contentView;
PendingIntent cOntentIntent= PendingIntent.getActivity(this, 0,
getIntent(), 0);
//notifcation.flags = Notification.FLAG_NO_CLEAR;// 永久在通知栏里
notification.flags = Notification.FLAG_AUTO_CANCEL;
//使用自定义下拉视图时,不需要再调用setLatestEventInfo()方法,但是必须定义contentIntent
notification.cOntentIntent= contentIntent;
notificationManager.notify(123, notification);//123是notification的唯一标识(id)

}


@Override
protected void onPause() {
super.onPause();
}

@Override
protected void onResume() {
super.onResume();

AppControl.JUDGE_DESTORY_ONCREATE += 1;
initSongData();
setPlayData(songListPosition);
}

@Override
public void onStart() {

super.onStart();
mainReceiver = new MainReceiver();
IntentFilter filter = new IntentFilter();
filter.addAction(SONG_STATE_UPDATE);
filter.addAction(MUSIC_CURRENT);
filter.addAction(MUSIC_DURATION);
filter.addAction(SONG_STATE_PRESS);
filter.addAction(AppControl.SERVICE_STOP);
registerReceiver(mainReceiver, filter);


}

private void initSongData() {
PlaySong playSOng= DataSupport.find(PlaySong.class, 1);
isPlayingBtn = playSong.isPlaying();//获得播放界面返回来的按键boolean值
long durate = playSong.getDuration();
int current = playSong.getCurrentTime();
int position = playSong.getSongPosition();
if (durate != 0) {
duration = durate;
}
if (current != 0) {
currentTime = current;
}
if (position != -1) {
sOngListPosition= position;
}
if (AppControl.JUDGE_DESTORY_OnCREATE== 0) {
isPlayingBtn = false;
}

if (isPlayingBtn) {
mainPlaySong.setBackgroundResource(R.drawable.play_normal);
} else {
mainPlaySong.setBackgroundResource(R.drawable.play_pressed);
}
}

private void setAdapter() {
//读取图片线程开启
songBitmapThread.run();
if (songList.size() == 0) {
Toast.makeText(MainActivity.this, "您的手机暂无歌曲信息", Toast.LENGTH_SHORT).show();
} else {
//配置歌曲列表适配器
sOngListAdapter= new SongListAdapter(MainActivity.this, songList);
songListView.setAdapter(songListAdapter);
/**
* 这里我要详细的说一下,
* 这里我主要实现了这样一个功能:比如我点击了歌曲A,进入了播放界面,
* 然后点击返回退出了播放界面进入列表界面,我再点击歌曲A,他不会重头开始播放,
* 然后我再点击B歌曲,他开始重头播放B。
*
* 不过这里我传给播放界面的参数我自己感觉有点繁琐,你们可以自己试着省略下
*/

songListView.setOnItemClickListener(new AdapterView.OnItemClickListener() {
@Override
public void onItemClick(AdapterView parent, View view, int position, long id) {
//点击歌曲进入播放界面
intent = new Intent(MainActivity.this, MusicPlayActivity.class);
/**
* songListPosition 这个表示的是当前正在播放的歌曲position
* 如果,两者相同,那就穿一些后面播放界面需要显示的状态过去,比如当前播放状态,
* 让播放界面的图标能够更新
*/

if (position == songListPosition) {
intent.putExtra("mainCurrentTime", currentTime);
intent.putExtra("isPlaying", isPlayingBtn);
intent.putExtra("MSSG", AppControl.MEDIA_PLAYING_SONG_IS_PLAYING);
intent.putExtra("isOncreate", AppControl.JUDGE_DESTORY_ONCREATE);

} else {
/**
*如果两个值不一样,证明你重新点击了一首歌
* MSSG 传的就和上面不一样了,使后面的播放界面做不同的处理
*/

intent.putExtra("isPlaying", isPlayingBtn);
intent.putExtra("mainCurrentTime", currentTime);
intent.putExtra("isPlaying", true);
intent.putExtra("MSSG", AppControl.MEDIA_PLAYING_SONG_AT_LIST);

}
/**
*
*
*/

intent.putExtra("position", position);
intent.putExtra("duration", songList.get(position).getDuration());
//这里当初图省事,这个传的是歌曲的循环状态,我当初写了个2,就是顺序播放,这个你们需要的可以自己改
intent.putExtra("repeatState", 2);
intent.putExtra("url", songList.get(position).getData());
sOngListPosition= position;
startActivity(intent);
//这个是主页面下面播放栏状态的更换
setPlayData(position);

}
});
}

}

@Override
public void onComplete(Platform platform, int i, HashMap hashMap) {

}

@Override
public void onError(Platform platform, int i, Throwable throwable) {

}

@Override
public void onCancel(Platform platform, int i) {

}

/**
* 开线程存图片
*/

private class SongBitmapThread implements Runnable {
@Override
public void run() {
sOngList= MusicLoader.getInstance(getApplicationContext().getContentResolver()).queryData();
for (int i = 0; i Bitmap bitmap = getBlurBitmap(MusicBitmap.getArtwork(MainActivity.this, songList.get(i).getId(), songList.get(i).getAlbumId(), true, true), MainActivity.this);
songList.get(i).setBitmap(bitmap);
}
}
}

/**
* 设置点击播放时下面播放栏的数据
*/

private void setPlayData(int position) {
songTitlePlay.setText(songList.get(position).getTitle());
songTitleArt.setText(songList.get(position).getArtist());
LogUtil.itSelf.Log("setPlayData" + songList.get(position).getTitle() + songList.get(position).getBitmap());
if (songList.get(position).getBitmap() == null) {

Resources res = getResources();
Bitmap bmp = BitmapFactory.decodeResource(res, R.drawable.morenbgsamall);
imSongPlaying.setImageBitmap(bmp);
} else {
imSongPlaying.setImageBitmap(songList.get(position).getBitmap());
}

}

public static Bitmap getBlurBitmap(Bitmap rootView, Context context) {
try {
if (rootView == null || cOntext== null) {
return null;
}
Bitmap bgBitmap = Bitmap.createBitmap(rootView);
return GetRoundBitmap.getInstance().getOvalBitmap(bgBitmap);
} catch (Exception e) {
e.printStackTrace();
}
return null;
}

private void createView() {
//歌曲列表
sOngList= new ArrayList();
//列表下面那个歌曲显示界面,有图片个按键那个linearlayout
linMainButtom = (LinearLayout) findViewById(R.id.lin_main_buttom);
//控制播放按键
mainPlaySOng= (Button) findViewById(R.id.main_play);
//歌曲列表listveiw
sOngListView= (ListView) findViewById(R.id.song_list);
sOngTitlePlay= (TextView) findViewById(R.id.song_title_play);
sOngTitleArt= (TextView) findViewById(R.id.song_album_play);
imSOngPlaying= (ImageView) findViewById(R.id.im_song_playing);
linMainButtom.setOnClickListener(this);
mainPlaySong.setOnClickListener(this);


}

@Override
protected void onDestroy() {
super.onDestroy();

if(!isRealDestroy){
showNotification();
}
}


/**
* 下面三个按键的点击触发事件
* 这里别看下面这么一大段代码,其实你去掉那些intent就是一点
* 这里主要实现了两个功能
* 1.点击下面的播放栏,能够进入播放界面
* 2.点击按键能控制歌曲的暂停去播放
* 3.AppControl.JUDGE_DESTORY_ONCREATE 这个是一个信号
* 是为了控制当你用菜单栏的形式强行关掉程序时,进来程序的时候,这个点击播放栏界面
* 不但要能进入播放界面,还要让歌曲开始播放,所以加了这么一个控制后面服务开始的功能
* @param v
*/

@Override
public void onClick(View v) {
switch (v.getId()) {

case R.id.main_play:
if (isPlayingBtn) {
Intent intent = new Intent(this, MusicPlayService.class);
intent.setAction("com.verzqli.media.play");
intent.putExtra("MSG", AppControl.MEDIA_PLAYING_SONG_PAUSE);
intent.putExtra("isOncreate", AppControl.JUDGE_DESTORY_ONCREATE);
mainPlaySong.setBackgroundResource(R.drawable.play_pressed);
startService(intent);
isPlayingBtn = false;
} else {
if (AppControl.JUDGE_DESTORY_OnCREATE== 0) {
Intent intent = new Intent(this, MusicPlayService.class);
intent.setAction("com.verzqli.media.play");
intent.putExtra("position", songListPosition);
intent.putExtra("url", songList.get(songListPosition).getData());
intent.putExtra("title", songList.get(songListPosition).getTitle());
intent.putExtra("MSG", AppControl.MEDIA_PLAYING_FIRST_PRESS);
intent.putExtra("progress", 0);
startService(intent);
mainPlaySong.setBackgroundResource(R.drawable.play_normal);
isPlayingBtn = true;
} else {
Intent intent = new Intent(this, MusicPlayService.class);
intent.setAction("com.verzqli.media.play");
intent.putExtra("MSG", AppControl.MEDIA_PLAYING_SONG_CONTINUE);
mainPlaySong.setBackgroundResource(R.drawable.play_normal);
isPlayingBtn = true;
startService(intent);
}
}
values.put("playing", isPlayingBtn);
DataSupport.update(PlaySong.class, values, 1);
break;
case R.id.lin_main_buttom:
intent = new Intent(MainActivity.this, MusicPlayActivity.class);
LogUtil.itSelf.Log("lin_main_buttom" + currentTime);
intent.putExtra("mainCurrentTime", currentTime);
intent.putExtra("isPlaying", isPlayingBtn);
intent.putExtra("MSSG", AppControl.MEDIA_PLAYING_SONG_IS_PLAYING);
intent.putExtra("position", songListPosition);
intent.putExtra("duration", songList.get(songListPosition).getDuration());
intent.putExtra("repeatState", 2);
intent.putExtra("url", songList.get(songListPosition).getData());
intent.putExtra("isOncreate", AppControl.JUDGE_DESTORY_ONCREATE);
startActivity(intent);
break;

}


}

/**
* 广播接收器
* 接受后台服务传过来的信息来更新界面
* 这里没什么,不懂的话可以评论留言,我会回答的
*/

private class MainReceiver 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);
} else if (action.equals(MUSIC_DURATION)) {
duration = intent.getIntExtra("duration", -1);
} else if (action.equals(SONG_STATE_UPDATE)) { //获取Intent中的current消息,current代表当前正在播放的歌曲

sOngListPosition= intent.getIntExtra("currentSongPosition", -1);

setPlayData(songListPosition);
}
if (action.equals(SONG_STATE_PRESS)) {
isPlayingBtn = intent.getBooleanExtra("isPlaying", true);
if (isPlayingBtn) {

mainPlaySong.setBackgroundResource(R.drawable.play_normal);
} else {
mainPlaySong.setBackgroundResource(R.drawable.play_pressed);
}
}if(action.equals(AppControl.SERVICE_STOP)) {
unregisterReceiver(mainReceiver);
Intent service = new Intent(MainActivity.this,MusicPlayService.class);
stopService(service);
isRealDestroy = true;

}
}


}

@Override
public boolean onCreateOptionsMenu(Menu menu) {
getMenuInflater().inflate(R.menu.menu_main, menu);

return super.onCreateOptionsMenu(menu);
}
}



推荐阅读
  • 深入解析Android自定义View面试题
    本文探讨了Android Launcher开发中自定义View的重要性,并通过一道经典的面试题,帮助开发者更好地理解自定义View的实现细节。文章不仅涵盖了基础知识,还提供了实际操作建议。 ... [详细]
  • 优化ListView性能
    本文深入探讨了如何通过多种技术手段优化ListView的性能,包括视图复用、ViewHolder模式、分批加载数据、图片优化及内存管理等。这些方法能够显著提升应用的响应速度和用户体验。 ... [详细]
  • 2023年京东Android面试真题解析与经验分享
    本文由一位拥有6年Android开发经验的工程师撰写,详细解析了京东面试中常见的技术问题。涵盖引用传递、Handler机制、ListView优化、多线程控制及ANR处理等核心知识点。 ... [详细]
  • 深入理解OAuth认证机制
    本文介绍了OAuth认证协议的核心概念及其工作原理。OAuth是一种开放标准,旨在为第三方应用提供安全的用户资源访问授权,同时确保用户的账户信息(如用户名和密码)不会暴露给第三方。 ... [详细]
  • 本文介绍如何通过注册表编辑器自定义和优化Windows文件右键菜单,包括删除不需要的菜单项、添加绿色版或非安装版软件以及将特定应用程序(如Sublime Text)添加到右键菜单中。 ... [详细]
  • 苹果新专利或将引领无边框手机时代
    苹果公司最近公布了一项新的专利技术,该技术能够在设备屏幕中嵌入光线传感器,这标志着苹果在实现无边框手机设计上迈出了重要一步。这一创新将极大提升手机的屏占比,并可能为未来的iPhone带来革命性的变化。 ... [详细]
  • 本文介绍如何使用 Python 提取和替换 .docx 文件中的图片。.docx 文件本质上是压缩文件,通过解压可以访问其中的图片资源。此外,我们还将探讨使用第三方库 docx 的方法来简化这一过程。 ... [详细]
  • RT,个人博客图片管理(方便管理,大家 ... [详细]
  • 网络运维工程师负责确保企业IT基础设施的稳定运行,保障业务连续性和数据安全。他们需要具备多种技能,包括搭建和维护网络环境、监控系统性能、处理突发事件等。本文将探讨网络运维工程师的职业前景及其平均薪酬水平。 ... [详细]
  • 从零开始构建完整手机站:Vue CLI 3 实战指南(第一部分)
    本系列教程将引导您使用 Vue CLI 3 构建一个功能齐全的移动应用。我们将深入探讨项目中涉及的每一个知识点,并确保这些内容与实际工作中的需求紧密结合。 ... [详细]
  • 帝国CMS多图上传插件详解及使用指南
    本文介绍了一款用于帝国CMS的多图上传插件,该插件通过Flash技术实现批量图片上传功能,显著提升了多图上传效率。文章详细说明了插件的安装、配置和使用方法。 ... [详细]
  • PHP 5.5.0rc1 发布:深入解析 Zend OPcache
    2013年5月9日,PHP官方发布了PHP 5.5.0rc1和PHP 5.4.15正式版,这两个版本均支持64位环境。本文将详细介绍Zend OPcache的功能及其在Windows环境下的配置与测试。 ... [详细]
  • 百度服务再次遭遇技术问题,疑似DNS解析故障
    近日晚间,百度多项在线服务出现加载异常,包括移动端搜索在内的多个功能受到影响。初步迹象表明,问题可能与DNS服务器解析有关。 ... [详细]
  • 本文详细介绍了《问道》手游在2020年12月31日进行的服务器维护情况,以及此次更新中新增的跨年狂欢活动和寒假活动等内容。同时,文章还涵盖了其他重要的系统优化与修复信息。 ... [详细]
  • Win11扩展卷无法使用?解决扩展卷灰色问题的指南
    本文详细介绍了在Windows 11中遇到扩展卷灰色无法使用时的解决方案,帮助用户快速恢复磁盘扩展功能。 ... [详细]
author-avatar
sdfa2255204
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有