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,点击上一首、下一首