Android应用界面开发_学习笔记_第三周
2016-03-13 15:56
441 查看
Fragment
一、什么是Fragment?1、Fragment是activity的界面中的一部分;多个Fragment们组合到一个activity中;多个activity中可重用一个Fragment。即Fragment相当于模块化的一段activity;
2、具有自己的生命周期,接收自己的事件;
3、在activity运行时被添加或删除。
二、为什么要使用Fragment?
1、支持更动态灵活的界面设计;2、activity的layout分成Fragment。
三、如何使用Fragment?
1、Create Fragment类 — onCreate() onCreateView() onPause()
2、Add Fragment
(1) Layout
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:orientation="vertical" android:layout_width="match_parent" android:layout_height="match_parent"> <fragment android:id="@+id/fragment_test" android:layout_width="match_parent" android:layout_height="200dp" android:name="com.example.chenjinhua.redcircle.FragmentTest" ></fragment> </LinearLayout>
activity_fragment.xml布局文件里添加如上代码后,运行,报如下错误:非法状态异常,Fragment没有创建View。
解决方法:
首先新建一个view布局文件如之前第二周写的item_listview.xml:
<?xml version="1.0" encoding="utf-8"?> <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent" android:layout_marginTop="5dp"> <ImageView android:id="@+id/avatar_imageview" android:layout_width="50dp" android:layout_height="40dp" /> <TextView android:id="@+id/item_name" android:layout_width="match_parent" android:layout_height="wrap_content" android:text="name" android:layout_toRightOf="@+id/avatar_imageview"/> <TextView android:id="@+id/item_age" android:layout_width="match_parent" android:layout_height="wrap_content" android:text="age" android:layout_below="@+id/item_name" android:layout_toRightOf="@+id/avatar_imageview"/> </RelativeLayout>
然后在FragmentTest类的onCreateView()方法里获取view:
@Override public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) { View view = inflater.inflate(R.layout.item_listview,container); return view; }
(2)Java Code
FragmentManager;
FragmentTransaction;
Add/Remove
FragmentTestActivity类
package com.example.chenjinhua.redcircle; import android.app.Activity; import android.app.FragmentManager; import android.app.FragmentTransaction; import android.os.Bundle; /** * Created by chenjinhua on 16/3/19. */ public class FragmentTestActivity extends Activity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_fragment); FragmentManager fragmentManager = getFragmentManager(); FragmentTransaction fragmentTransaction = fragmentManager.beginTransaction(); FragmentTest fragmentTest = new FragmentTest(); fragmentTransaction.add(R.id.fragment_test_view,fragmentTest).commit(); } }
activity_fragment.xml布局文件:
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:orientation="vertical" android:background="@color/red" android:layout_width="match_parent" android:layout_height="match_parent"> <fragment android:id="@+id/fragment_test" android:layout_width="match_parent" android:layout_height="wrap_content" android:name="com.example.chenjinhua.redcircle.FragmentTest" ></fragment> <RelativeLayout android:layout_marginTop="100dp" android:id="@+id/fragment_test_view" android:layout_width="match_parent" android:layout_height="match_parent" /> </LinearLayout>
FragmentTest类 同上。
运行报如下错误:
FragmentTestActivity类里fragment_test_view是父view;FragmentTest类里item_listview是子view。
解决办法:修改FragmentTest类里的onCreateView()方法
@Override public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) { //解析出来的xml视图放到container里,不绑定 View view = inflater.inflate(R.layout.item_listview,container,false); return view; }
程序运行结果:
第一个Fragment是通过布局文件生成的,第二个Fragment是通过java code生成的。
删除fragment:修改FragmentTestActivity类代码为d
fragmentTransaction.add(R.id.fragment_test_view,fragmentTest); fragmentTransaction.remove(fragmentTest).commit();
四、如何管理Fragment?
1、查找Fragment — findFragmentById() 和 findFragmentByTag()
Fragment fragment = fragmentManager.findFragmentById(R.id.fragment_test); if (fragment instanceof FragmentTest){ // TODO: Do your Action. }else{ throw new IllegalStateException("iis not fragmentTest"); }
java里面的二元运算符,判断左边的对象是否是右边类的实例。假如是的话,返回true;假如不是的话,返回false
2、Fragment的后退 — Fragment Stack; popBackStack(); addOnBackStackChangedListener()
五、Fragment 生命周期
参考博客:
http://blog.csdn.net/forever_crying/article/details/8238863/
Handler
多线程与异步Handler用来做什么?
1、定时执行Message和MessageQueue;
2、在不同线程中执行Runnable。
Handler怎么使用?
obtainMessage() //取得消息
sendMessage() //发送消息
handlerMessage() //处理消息
Message、 MessageQueue、Looper
Message 有2个整型数值 和 1个Object;
MessageQueue是Message的队列
Looper消息泵
Looper的原理
如何使用:
1、是MessageQueue的管理者;
2、Looper.prepare()
3、每一个Looper对象和一个线程关联
4、Looper.myLooper()可以获得当前线程的Looper对象
Looper从MessageQueue中取出Message,交由Handler的handlerMessage进行处理;调用Message.recycle()将其放入Message Pool中。
自定义控件、自定义属性、画视图
一、提取布局属性 theme & style
把视图控件中大部分相同的属性抽取出来1、Theme是针对窗体级别的,改变窗体样式;Style 是针对窗体元素级别的,改变指定控件或者Layout的样式;
2、抽象view的共同属性;
3、可继承
使用步骤:
1、创建属性; 2、布局中设置。
1、创建属性
比如有3个TextView,他们有一些相同的属性;
在res - values - styles.xml里创建属性。
2、布局中设置
style.xml里子控件属性可以继承父控件属性,子控件的属性会覆盖父控件的属性。
同理,Theme也是,在styles.xml里创建theme;在Manifest里设置。
如下图设置了整个application的theme,也可以对每个activity设置theme。
三、View是如何工作的? – 自定义控件
1、构造器 —> 初始化2、onMesure() 定大小
3、onLayout() 定位置
4、onDraw() 绘制
5、invalidate() 刷新
自定义控件的三种主要创建形式
1、继承已有的控件来实现自定义控件,如 Button、 CheckedTextView;
2、通过继承一个布局文件实现自定义控件,如RelativeLayout;
3、通过继承view类来实现自定义控件。
栗子:做一个简单的自定义控件
1、做一个圆形的红色按钮;
2、中间有一个白色的数字;
3、数字起始为20;
4、每点击一次减少1。
1、代码里onDraw() - 自定义控件;
2、视图xml文件里引用自定义控件。
DrawRedCircleView类
package com.example.chenjinhua.redcircle; import android.content.Context; import android.graphics.Canvas; import android.graphics.Color; import android.graphics.Paint; import android.graphics.Rect; import android.util.AttributeSet; import android.view.View; public class DrawRedCircleView extends View implements View.OnClickListener{ private Paint mPaint; private Rect mRect; private int mNumber = 20; public DrawRedCircleView(Context context) { this(context, null); } public DrawRedCircleView(Context context, AttributeSet attrs) { this(context, attrs, 0); } public DrawRedCircleView(Context context, AttributeSet attrs, int defStyleAttr) { super(context, attrs, defStyleAttr); init(); } private void init() { mPaint = new Paint(); mRect = new Rect(); } @Override protected void onDraw(Canvas canvas) { super.onDraw(canvas); mPaint.setColor(Color.RED); canvas.drawCircle(getWidth() / 2, getWidth() / 2, getWidth() / 2, mPaint); mPaint.setColor(Color.WHITE); mPaint.setTextSize(100); String textNumber = String.valueOf(mNumber); mPaint.getTextBounds(textNumber,0,textNumber.length(),mRect); int textWidth = mRect.width(); int textHight = mRect.height(); canvas.drawText(textNumber,getWidth()/2 - textWidth/2,getWidth()/2 + textHight/2,mPaint); View view = findViewById(R.id.redCircle_view); view.setOnClickListener(this); } @Override public void onClick(View view) { if ( mNumber > 0 ){ mNumber --; }else{ mNumber = 20; } invalidate(); } }
RedCircleActivity类
package com.example.chenjinhua.redcircle; import android.app.Activity; import android.os.Bundle; /** * Created by chenjinhua on 16/3/15. */ public class RedCircleActivity extends Activity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_redcircle); } }
MainActivity类
package com.example.chenjinhua.redcircle; import android.app.Activity; import android.content.Intent; import android.os.Bundle; import android.view.View; import android.widget.TextView; /** * Created by chenjinhua on 16/3/15. */ public class MainActivity extends Activity implements View.OnClickListener{ @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); TextView textView_mainActivity = (TextView) findViewById(R.id.textView_mainActivity); textView_mainActivity.setOnClickListener(this); } @Override public void onClick(View view) { startActivity(new Intent(MainActivity.this, RedCircleActivity.class)); } }
activity_redcircle.xml
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:orientation="vertical" android:layout_width="match_parent" android:layout_height="match_parent"> <com.example.chenjinhua.redcircle.DrawRedCircleView android:id="@+id/redCircle_view" android:layout_width="200dp" android:layout_height="200dp"/> </LinearLayout>
如何自定义属性
对于自定义属性,遵循以下几步,就可以实现:1、自定义一个CustomView(extends View )类
2、编写values/attrs.xml,在其中编写styleable和item等标签元素
3、在布局文件中CustomView使用自定义的属性(注意namespace)
4、在CustomView的构造方法中通过TypedArray获取
在哪里 创建属性 ?
res - values 下新建xml文件,如attrs.xml。
attrs.xml :
<?xml version="1.0" encoding="utf-8"?> <resources> //类 <declare-styleable name="ViewTest"> //每一个attr都是一个属性,每个属性都有一个format <attr name="backgroundColor" format="color"/> <attr name="textColor" format="color"/> <attr name="textSize" format="dimension"/> </declare-styleable> </resources>
如何使用这些属性:
代码中引用
private int mBackgroundColor; //根据名字读取attr.xml里的属性(是一个数组) TypedArray typedArray = context.obtainStyledAttributes(attrs,R.styleable.ViewTest); //从读取出来的数组,获取BackgroundColor的属性,没有的话默认值为RED给变量mBackgroundColor。 mBackgroundColor = typedArray.getColor(R.styleable.ViewTest_backgroundColor,Color.BLUE); mPaint.setColor(mBackgroundColor);
布局中设置 - activity_draw.xml:
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" //添加自定义命名空间 xmlns:jinhua="http://schemas.android.com/apk/res-auto" android:orientation="vertical" android:layout_width="match_parent" android:layout_height="match_parent"> <!--引用自定义的控件--> <com.example.chenjinhua.thirdweek.ViewTest android:layout_width="300dp" android:layout_height="300dp" jinhua:backgroundColor="@color/colorAccent" jinhua:textSize="18dp"/> </LinearLayout>
小技巧
1、参考系统源码的实现2、多了解原理
Tips
1、什么时候需要使用init()方法init : 单词翻译过来是 初始化的意思, 不是一个官方提供的api函数,程序员习惯上 把 要用的控件 都在 oncreate中 统一实例化, 但是把代码放在oncreate中很难看, 所以再创一个init( ) 函数 ……之类的,把初始化代码(new出来的对象等)放里面,方便整改。
2、利用画布绘制文字
mPaint.setColor(Color.WHITE); mPaint.setTextSize(100); //要写的数字 String text = String.valueOf(mNumber); //获取文字四周矩形的边距,放到mRect对象里。 mPaint.getTextBounds(text,0,text.length(),mRect); int textWidth = mRect.width(); int textHight = mRect.height(); canvas.drawText(text,getWidth()/2 - textWidth/2,getHeight()/2 + textHight/2,mPaint);
3、dp、sp 和 px
是什么?
px:像素点
dp:与像素密度密切相关 (100px 在不同手机上大小显示不一致,dp就解决了这个问题)
sp:相当于dp(用来修饰文字)
dip:= dp
怎么用?
文字的尺寸一律用sp单位;
非文字的尺寸一律用dp单位;
偶尔需要使用px单位:如需要在屏幕上画一条细的分割线 1px。
相关文章推荐
- 安卓(android)建立项目时失败,出现Android Manifest.xml file missing几种解决方法?(总结中)
- 设计模式_模板方法模式
- Android-Sample1-NetWork
- Android开发之在代码中给控件设置颜色不起作用的解决办法
- 01.android设置应用开机启动
- Android事件分发机制源码剖析(2)—顶层View对点击事件的分发过程
- Android为什么使用Binder-android学习之旅(101)
- Android为什么使用Binder-android学习之旅(101)
- Android为什么使用Binder-android学习之旅(101)
- 基于Proxy思想的Android插件框架
- Android 项目代码结构框架图。
- Android Studio--EditText属性
- 自学android课程 Android 学习Activity(1)activity和intent
- 一些Android Demo的网址
- 作为一个IOS\android开发者,转行到RN的理由,包括学习路线
- Android项目中图标的更改
- android权限大全
- 闹铃设置
- Android的Binder的起源-android学习之旅(100)
- Android的Binder的起源-android学习之旅(100)