国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁(yè) > 系統(tǒng) > Android > 正文

Android 自定義控件詳解及實(shí)例代碼

2019-12-12 03:23:22
字體:
來(lái)源:轉(zhuǎn)載
供稿:網(wǎng)友

開(kāi)發(fā)自定義控件的步驟:

1、了解View的工作原理
2、 編寫(xiě)繼承自View的子類(lèi)
3、 為自定義View類(lèi)增加屬性
4、 繪制控件
5、 響應(yīng)用戶(hù)消息
6 、自定義回調(diào)函數(shù)  

一、View結(jié)構(gòu)原理

Android系統(tǒng)的視圖結(jié)構(gòu)的設(shè)計(jì)也采用了組合模式,即View作為所有圖形的基類(lèi),Viewgroup對(duì)View繼承擴(kuò)展為視圖容器類(lèi)。
View定義了繪圖的基本操作
基本操作由三個(gè)函數(shù)完成:measure()、layout()、draw(),其內(nèi)部又分別包含了onMeasure()、onLayout()、onDraw()三個(gè)子方法。具體操作如下:

1、measure操作

     measure操作主要用于計(jì)算視圖的大小,即視圖的寬度和長(zhǎng)度。在view中定義為final類(lèi)型,要求子類(lèi)不能修改。measure()函數(shù)中又會(huì)調(diào)用下面的函數(shù):

     (1)onMeasure(),視圖大小的將在這里最終確定,也就是說(shuō)measure只是對(duì)onMeasure的一個(gè)包裝,子類(lèi)可以覆寫(xiě)onMeasure()方法實(shí)現(xiàn)自己的計(jì)算視圖大小的方式,并通過(guò)setMeasuredDimension(width, height)保存計(jì)算結(jié)果。 

2、layout操作

     layout操作用于設(shè)置視圖在屏幕中顯示的位置。在view中定義為final類(lèi)型,要求子類(lèi)不能修改。layout()函數(shù)中有兩個(gè)基本操作:

     (1)setFrame(l,t,r,b),l,t,r,b即子視圖在父視圖中的具體位置,該函數(shù)用于將這些參數(shù)保存起來(lái);
     (2)onLayout(),在View中這個(gè)函數(shù)什么都不會(huì)做,提供該函數(shù)主要是為viewGroup類(lèi)型布局子視圖用的; 

3、draw操作

     draw操作利用前兩部得到的參數(shù),將視圖顯示在屏幕上,到這里也就完成了整個(gè)的視圖繪制工作。子類(lèi)也不應(yīng)該修改該方法,因?yàn)槠鋬?nèi)部定義了繪圖的基本操作:

     (1)繪制背景;
     (2)如果要視圖顯示漸變框,這里會(huì)做一些準(zhǔn)備工作;
     (3)繪制視圖本身,即調(diào)用onDraw()函數(shù)。在view中onDraw()是個(gè)空函數(shù),也就是說(shuō)具體的視圖都要覆寫(xiě)該函數(shù)來(lái)實(shí)現(xiàn)自己的顯示(比如TextView在這里實(shí)現(xiàn)了繪制文字的過(guò)程)。而對(duì)于ViewGroup則不需要實(shí)現(xiàn)該函數(shù),因?yàn)樽鳛槿萜魇恰皼](méi)有內(nèi)容“的,其包含了多個(gè)子view,而子View已經(jīng)實(shí)現(xiàn)了自己的繪制方法,因此只需要告訴子view繪制自己就可以了,也就是下面的dispatchDraw()方法;
     (4)繪制子視圖,即dispatchDraw()函數(shù)。在view中這是個(gè)空函數(shù),具體的視圖不需要實(shí)現(xiàn)該方法,它是專(zhuān)門(mén)為容器類(lèi)準(zhǔn)備的,也就是容器類(lèi)必須實(shí)現(xiàn)該方法;
     (5)如果需要(應(yīng)用程序調(diào)用了setVerticalFadingEdge或者setHorizontalFadingEdge),開(kāi)始繪制漸變框;
     (6)繪制滾動(dòng)條;
      從上面可以看出自定義View需要最少覆寫(xiě)onMeasure()和onDraw()兩個(gè)方法。 

二、View類(lèi)的構(gòu)造方法

創(chuàng)建自定義控件的3種主要實(shí)現(xiàn)方式:

1)繼承已有的控件來(lái)實(shí)現(xiàn)自定義控件: 主要是當(dāng)要實(shí)現(xiàn)的控件和已有的控件在很多方面比較類(lèi)似, 通過(guò)對(duì)已有控件的擴(kuò)展來(lái)滿(mǎn)足要求。

2)通過(guò)繼承一個(gè)布局文件實(shí)現(xiàn)自定義控件,一般來(lái)說(shuō)做組合控件時(shí)可以通過(guò)這個(gè)方式來(lái)實(shí)現(xiàn)。

    注意此時(shí)不用onDraw方法,在構(gòu)造廣告中通過(guò)inflater加載自定義控件的布局文件,再addView(view),自定義控件的圖形界面就加載進(jìn)來(lái)了。

3)通過(guò)繼承view類(lèi)來(lái)實(shí)現(xiàn)自定義控件,使用GDI繪制出組件界面,一般無(wú)法通過(guò)上述兩種方式來(lái)實(shí)現(xiàn)時(shí)用該方式。 

三、自定義View增加屬性的兩種方法:

1)在View類(lèi)中定義。通過(guò)構(gòu)造函數(shù)中引入的AttributeSet 去查找XML布局的屬性名稱(chēng),然后找到它對(duì)應(yīng)引用的資源ID去找值。
案例:實(shí)現(xiàn)一個(gè)帶文字的圖片(圖片、文字是onDraw方法重繪實(shí)現(xiàn))

public class MyView extends View {    private String mtext;  private int msrc;  public MyView(Context context) {    super(context);  }  public MyView(Context context, AttributeSet attrs) {    super(context, attrs);    int resourceId = 0;    int textId = attrs.getAttributeResourceValue(null, "Text",0);    int srcId = attrs.getAttributeResourceValue(null, "Src", 0);    mtext = context.getResources().getText(textId).toString();    msrc = srcId;  }    @Override  protected void onDraw(Canvas canvas) {    Paint paint = new Paint();    paint.setColor(Color.RED);    InputStream is = getResources().openRawResource(msrc);         Bitmap mBitmap = BitmapFactory.decodeStream(is);        int bh = mBitmap.getHeight();        int bw = mBitmap.getWidth();      canvas.drawBitmap(mBitmap, 0,0, paint);    //canvas.drawCircle(40, 90, 15, paint);    canvas.drawText(mtext, bw/2, 30, paint);  }}

布局文件:

<?xml version="1.0" encoding="utf-8"?><LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"  android:layout_width="match_parent"  android:layout_height="match_parent"  android:orientation="vertical" >  <com.example.myimageview2.MyView    android:id="@+id/myView1"    android:layout_width="wrap_content"    android:layout_height="wrap_content"     Text="@string/hello_world"    Src="@drawable/xh"/></LinearLayout>

屬性Text, Src在自定義View類(lèi)的構(gòu)造方法中讀取。 

2)通過(guò)XML為View注冊(cè)屬性。與Android提供的標(biāo)準(zhǔn)屬性寫(xiě)法一樣。

案例:  實(shí)現(xiàn)一個(gè)帶文字說(shuō)明的ImageView (ImageView+TextView組合,文字說(shuō)明,可在布局文件中設(shè)置位置)

public class MyImageView extends LinearLayout {  public MyImageView(Context context) {    super(context);  }  public MyImageView(Context context, AttributeSet attrs) {    super(context, attrs);    int resourceId = -1;    TypedArray typedArray = context.obtainStyledAttributes(attrs,        R.styleable.MyImageView);    ImageView iv = new ImageView(context);    TextView tv = new TextView(context);    int N = typedArray.getIndexCount();    for (int i = 0; i < N; i++) {      int attr = typedArray.getIndex(i);      switch (attr) {      case R.styleable.MyImageView_Oriental:        resourceId = typedArray.getInt(            R.styleable.MyImageView_Oriental, 0);        this.setOrientation(resourceId == 1 ? LinearLayout.HORIZONTAL            : LinearLayout.VERTICAL);        break;      case R.styleable.MyImageView_Text:        resourceId = typedArray.getResourceId(            R.styleable.MyImageView_Text, 0);        tv.setText(resourceId > 0 ? typedArray.getResources().getText(            resourceId) : typedArray            .getString(R.styleable.MyImageView_Text));        break;      case R.styleable.MyImageView_Src:        resourceId = typedArray.getResourceId(            R.styleable.MyImageView_Src, 0);        iv.setImageResource(resourceId > 0 ?resourceId:R.drawable.ic_launcher);        break;        }    }    addView(iv);    addView(tv);    typedArray.recycle();  }}

attrs.xml進(jìn)行屬性聲明, 文件放在values目錄下

<?xml version="1.0" encoding="utf-8"?><resources>  <declare-styleable name="MyImageView">    <attr name="Text" format="reference|string"></attr>    <attr name="Oriental" >      <enum name="Horizontal" value="1"></enum>      <enum name="Vertical" value="0"></enum>    </attr>    <attr name="Src" format="reference|integer"></attr>  </declare-styleable></resources>

MainActivity的布局文件:先定義命名空間 xmlns:uview="http://schemas.android.com/apk/res/com.example.myimageview2" (com.example.myimageview2為你
在manifest中定義的包名)

然后可以像使用系統(tǒng)的屬性一樣使用:uview:Oriental="Vertical"

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"  xmlns:uview="http://schemas.android.com/apk/res/com.example.myimageview2"  xmlns:tools="http://schemas.android.com/tools"  android:layout_width="match_parent"  android:layout_height="match_parent"  android:orientation="vertical"  tools:context=".MainActivity" >  <TextView    android:layout_width="wrap_content"    android:layout_height="wrap_content"    android:text="@string/hello_world" />  <com.example.myimageview2.MyImageView    android:id="@+id/myImageView1"    android:layout_width="wrap_content"    android:layout_height="wrap_content"    uview:Text="這是一個(gè)圖片說(shuō)明"     uview:Src="@drawable/tw"    uview:Oriental="Vertical">  </com.example.myimageview2.MyImageView></LinearLayout>

四、控件繪制 onDraw() 

五、:自定義View的方法

onFinishInflate() 回調(diào)方法,當(dāng)應(yīng)用從XML加載該組件并用它構(gòu)建界面之后調(diào)用的方法onMeasure() 檢測(cè)View組件及其子組件的大小onLayout() 當(dāng)該組件需要分配其子組件的位置、大小時(shí)onSizeChange() 當(dāng)該組件的大小被改變時(shí)onDraw() 當(dāng)組件將要繪制它的內(nèi)容時(shí)onKeyDown 當(dāng)按下某個(gè)鍵盤(pán)時(shí)onKeyUp 當(dāng)松開(kāi)某個(gè)鍵盤(pán)時(shí)onTrackballEvent 當(dāng)發(fā)生軌跡球事件時(shí)onTouchEvent 當(dāng)發(fā)生觸屏事件時(shí)onWindowFocusChanged(boolean) 當(dāng)該組件得到、失去焦點(diǎn)時(shí)onAtrrachedToWindow() 當(dāng)把該組件放入到某個(gè)窗口時(shí)onDetachedFromWindow() 當(dāng)把該組件從某個(gè)窗口上分離時(shí)觸發(fā)的方法onWindowVisibilityChanged(int): 當(dāng)包含該組件的窗口的可見(jiàn)性發(fā)生改變時(shí)觸發(fā)的方法

感謝閱讀,希望能幫助到大家,謝謝大家對(duì)本站的支持!

發(fā)表評(píng)論 共有條評(píng)論
用戶(hù)名: 密碼:
驗(yàn)證碼: 匿名發(fā)表
主站蜘蛛池模板: 宜宾市| 滁州市| 三台县| 浙江省| 宜兰县| 巢湖市| 汕头市| 保山市| 咸丰县| 盐城市| 通海县| 苏尼特左旗| 读书| 望城县| 平武县| 喜德县| 洛浦县| 股票| 湟源县| 阿勒泰市| 乌兰浩特市| 宁安市| 牡丹江市| 邢台县| 榕江县| 烟台市| 突泉县| 元朗区| 新野县| 玉林市| 澎湖县| 偃师市| 襄汾县| 信阳市| 比如县| 军事| 平塘县| 瑞昌市| 常州市| 珲春市| 高淳县|