1,二级列表(ExpandableListView)
创建布局,找控件
1.创建布局,找控件
2.获取数据:网络数据,死数据
3.创建适配器:10个方法,两个优化(ViewHolder 避免重复找id,ContentView存放已创建的布局,方便复用)
4.设置适配器
5.父项,子项点击事件(——》前提如下图:)
2,ListView手动,自动加载更多
(1)listView
1.创建布局,找控件
2.获取网络数据
3.创建适配器
4.设置适配器
(2)手动加载更多
1.创建布局,找控件
2.获取网络数据
3.创建适配器
4.设置适配器
5.给listView添加一个footer
6.点击footer中button加载更多:page++,获取数据
(3)网络加载更多
1.创建布局,找控件
2.获取网络数据
3.创建适配器
4.设置适配器
5.定义一个变量isbuttom表示是否滑倒底部
6.listView设置滑动监听事件
SCROLL_STATE_IDLE,手指未触摸屏幕,且屏幕静止
SCROLL_STATE_TOUCH_SCROLL,手指未离开屏幕滑动
SCROLL_STATE_FLING,手指使劲滑动屏幕,然后手指离开屏幕,屏幕仍在滑动
3,侧滑菜单
1,侧滑菜单
1.添加依赖
2.创建布局:DrawerLayout,NavigationView(打开方式:layout_gravity=left,header,menu)
3.设置ActionBar 在styles修改属性为NoActionBar,toolbar设置logo,标题,副标题,关联toolbar和侧滑菜单
4.监听事件,侧滑头部监听(mNv.getHeaderView获取),侧滑菜单监听(mNv.setNavigationItemSelectedListener)
DrawerLayout(随X轴移动-mDl.addDrawerListener——getRight()),代码关闭打开侧滑菜单(openDrawer(Gravity.LEFT),closeDrawer(Gravity.LEFT))
5.沉浸状态栏,首先DrawerLayout引入属性android:fitsSystemWindows="true",接下来在res下创建文件values-v21,文件内添加文件styles.xml,添加属性#00FFFFFF
4,选项菜单
1,创建选项菜单 onCreateOptionsMenu:menu创建的两种方式(代码,menu)showAsAction显示在toolbar
2.选项菜单的点击事件 onOptionsItemSelected 通过switch选择
5,上下文菜单
1.注册上下文菜单 registerForContextMenu()
2.创建上下文菜单 onCreateContextMenu
3.上下文菜单的点击事件 onContextItemSelected
选项菜单和上下文菜单区别
OptionsMenu是整个界面共用,ContextMenu是注册给某个组件,此组件拥有菜单,没有注册的组件没此菜单
6,RecyclerView
一、recyclerView
1,添加依赖(版本问题注意),创建布局
2,设置布局管理器(三种显示方式:线性布局、网格布局、瀑布流布局)
3,获取数据
4,创建适配器——重写三个,通过接口回调实现点击事件
LinearLayoutManager()、GridLayoutManager()、StaggeredGridLayoutManager()
5设置适配器
二、RecyclerView布局:list+banner
1, RecyclerView基本使用
2,定义类型常量
3,重写方法getItemViewType()根据位置返回不同类型
4,重写oncreateViewHolder()根据不同类型加载不同的布局
5,重写onBindViewHolder()根据不用类型加载不同数据
注意
1、获取条目总数:list.size()+1
2、onBindViewHolder:获取别表条目数据的时候,position+1
7,RecyclerView通过接口回调实现点击事件
1,在adapter定义一个内部接口,内部接口定义一个方法,方法参数是我们需要返回值的;
2,在adapter定义接口变量,并设置set方法
3,在onBindVIewholder()中,给条目做一个点击事件
4,在fragment或者activity中,使用adapter对象调用点击事件即可
8,RecyclerView添加刷新
1,添加依赖
2,在布局中添加刷新的控件并找控件
3,给刷新控件添加加载更多、下拉刷新添加监听
4,刷新完毕列表,关闭SmartRefreshLayout头和脚。
9、fragment
静态添加fragment
1,创建一个fragment
2,创建布局,
注意:
必须要有id,否则:Caused by: java.lang.IllegalArgumentException: Binary XML file line #9: Must specify unique android:id, android:tag, or have a parent with an id for com.anfly.fragmentr.AFragment
布局中必须添加属性name,值该fragment全类名
动态添加fragment
//获取碎片管理器FragmentManager fm=getSupportFragmentManager();
//开启事务FragmentTransaction fragmentTransaction=fm.beginTransaction();
//获取fragment对象AFragment aFragment=newAFragment();
//替换容器中内容fragmentTransaction.replace(R.id.fl_container,aFragment);
//提交事务fragmentTransaction.commit();
Transaction常用方法
1,add 添加
2,remove 移除
3,replace 替换
4,hide 隐藏
5,show 显示
6,attach 重建view视图,附加到ui上并显示
7,detach 会将view视图从UI中移除,和remove()不同,此时fragment的状态依然由fragment的状态维护
8,commit 提交
fragment生命周期
1,onAttach() 当Fragment与Activity发生关联时调用。
2,onCreate()
3,onCreateView() 创建该Fragment的视图
4,onActivityCreated() 当Activity的onCreate方法返回时调用
5,onStart()
6,onResume()
7,onPause()
8,onStop()
9,onDestroyView() 当fragment中的视图被移除的时候,调用这个方法。
10,onDestroy()
11,onDetach() 当fragment和activity分离的时候,调用这个方法。
fragment传递数据到activity
1,获取activity对象,直接调用方法
MainActivity activity = (MainActivity) getActivity();activity.getMsgFromFramgent("我是来自fragment的数据");
2,接口回调传递数据
3,通过fragment的有参构造传数据(不推荐)
activity传递数据到fragment
通过bundle方式传值
activity中:
AFragment aFragment=newAFragment();Bundle bundle=newBundle();bundle.putString("a","我是来自activity的数据");aFragment.setArguments(bundle);
fragment中:
Bundle bundle=getArguments();String a=bundle.getString("a");
fragment与fragment之间传递数据
1,通过构造方式传值(不推荐)
2,通过FragmentManager找到对应Id或者Tag的Framgment,然后获取里面的数据或方法
3,通过它们所在的Activity作为桥梁,可以使用getActivity()或者接口回调,达到获取另一个Fragment数据的目的.
6.ViewPager+tablayout
1》ViewPager结合view实现导航
①创建布局找控件
②获取数据集合
③创建适配器:
getCount()isViewFromObject() instantiateItem()destroyItem()
④设置适配器
2》Viewpager结合Fragment实现导航
①创建布局找控件
②获取fragment的集合fragments
③创建适配器:FragmentStatePagerAdapter和FragmentPagerAdapter区别以及内部方法
④设置适配器
3》Banner开源框架
banner.setBannerStyle(BannerConfig.NUM_INDICATOR_TITLE)//设置风格
.setImages(images)//设置图片集合
.setBannerAnimation(Transformer.DepthPage)//设置动画
.setBannerTitles(titles)//直接添加无效,必须设置BannerStyle
.setImageLoader(newGlideImageLoader())//图片加载器
.start();
4》Tablayout
①属性
5》TVF(TabLayout+ViewPage+Fragment)
①创建布局找控件:TV
②创建两个集合:fragments和titles
③创建适配器:四个方法(包含一个构造)
④设置适配器
⑤TV结合:tab.setupWithViewPager(vp);
⑥设置图片选择器tab.getTabAt(0).setIcon()
Version:1.0StartHTML:000000201EndHTML:000022817StartFragment:000008969EndFragment:000022779StartSelection:000008969EndSelection:000022779SourceURL:https://www.jianshu.com/p/fe890d03854c
7.PopupWindow
①创建PopupWindow布局
②创建PopupWindow对象,用三个参数的构造
③PopupWindow四种显示方式
④聚焦:EditText能输入内容
⑤点击范围外关闭PopupWindow
⑥全屏阴影,PopupWindow点击消失监听
⑦进出场动画
8.Notification
②兼容O版以上系统
③获取通知对象(构建者模式):必要属性有三项
④用通知管理器发送通知
⑤延时意图:intent、pendingIntent、setContentIntent
⑥通知提示:声音、震动、呼吸灯、全部
9.权限
分类:
①普通权限:不需要动态获取
②危险权限:需要动态获取
危险权限分类:3CSLMP
如何动态获取权限:
①在清单列表写上需要的全下你
②检查是否授权
如果授权 -> 操作
如果没有授权 -> 请求权限
if(ActivityCompat.checkSelfPermission(this,Manifest.permission.CALL_PHONE)==PackageManager.PERMISSION_GRANTED){callPhone();}else{ActivityCompat.requestPermissions(this,newString[]{Manifest.permission.CALL_PHONE},100);}
③请求权限结果
@OverridepublicvoidonRequestPermissionsResult(int requestCode,@NonNullString[]permissions,@NonNullint[]grantResults){super.onRequestPermissionsResult(requestCode,permissions,grantResults);switch(requestCode){case100:if(grantResults.length>0&&grantResults[0]==PackageManager.PERMISSION_GRANTED){callPhone();}else{Toast.makeText(MainActivity.this,"授权失败",Toast.LENGTH_SHORT).show();}break;}}
通过框架获取危险权限:
①添加依赖implementation 'com.github.dfqin:grantor:2.5'
②使用
PermissionsUtil.requestPermission(this,newPermissionListener(){@OverridepublicvoidpermissionGranted(@NonNullString[]permission){callPhone();}@OverridepublicvoidpermissionDenied(@NonNullString[]permission){Toast.makeText(MainActivity.this,"授权失败",Toast.LENGTH_SHORT).show();}},Manifest.permission.CALL_PHONE);
十、内容提供者
ContentProvider
①创建一个数据库及一张表
②自定义ContentProvider继承自ContentProvider,重写方法
ContentResolver
①获取ContentResolver
②获取uri:Uri.parse("content://"+authorities+/+path)
Uri uri=Uri.parse("content://com.example.contentprovider.ClContentProvider/cl");
ContentResolver读取短信、通讯录、图片、音频、视频
1,在清单列表里添加权限
2,动态获取危险权限
3,获取ContentResolver对象contentResolver
4,contentResolver调用query()方法查询相关内容
短信:Telephony.Sms.CONTENT_URI
通讯录:ContactsContract.CommonDataKinds.Phone.CONTENT_URI
图片:MediaStore.Images.Media.EXTERNAL_CONTENT_URI
音频:MediaStore.Audio.Media.EXTERNAL_CONTENT_URI
视频:MediaStore.Video.Media.EXTERNAL_CONTENT_URI
Service
startService生命周期
onCreate()(开启服务)-->onStartCommand()(服务运行)-->onDestory()(停止服务)
bindService生命周期
onCreate()(开启服务)-->onBind()(绑定服务)-->onUnbind(解绑服务)-->onDestory(停止服务)
startService和bindService区别
1.启动服务中使用StartService()方法来进行方法的调用,调用者和服务之间没有联系,即使调用者退出了,服务依然在进行,onCreate()- >onStartCommand()->onDestroy(),注意其中没有onStart(),主要是被onStartCommand()方法给取代了,onStart方法不推荐使用了。
2.绑定服务中使用bindService()方法来绑定服务,调用者和绑定者绑在一起,调用者一旦退出服务也就终止了onCreate()->onBind()->onUnbind()->onDestroy()。
Activity和Service之间的数据传递
1,数据从Activity和到Service
intent方式:startService和bindService都可以
2,数据从Activity到Service
IBinder方式:bindService
3,数据从Service到Activity
接口回调和广播
音乐播放器
MediaPlayer创建方式
1,MediaPlayer mp = new MediaPlayer();
2,MediaPlayer mp = MediaPlayer.create(this, R.raw.test);
四种资源
1,用户在应用中事先自带的resource资源
2,存储在SD卡或其他文件路径下的媒体文件
3,网络上的媒体文件
4,assets目录下文件
assets 和 raw 资源文件夹区别
结合SeekBar实现拖动播放音乐功能
1,创建seekbar布局
2,seekbar设置监听,在停止拖动中player.seekTo(seekBar.getProgress());
3,更新seekbar:
结合RecyclerView实现音乐播放上一首,下一首功能
1,使用ContentResolver+recyclerview展示音乐列表
2,条目点击事件播放对应的音乐
3,点击上一首、下一首