1. 程式人生 > >Android自定義View高階(一)-分類與流程

Android自定義View高階(一)-分類與流程

一.自定義View繪製流程

Paste_Image.png

二.自定義View分類

  • 自定義ViewGroup
    自定義ViewGroup一般是利用現有的元件根據特定的佈局方式來組成新的元件,大多繼承自ViewGroup或各種Layout。

  • 自定義View
    在沒有現成的View,需要自己實現的時候,就使用自定義View,一般繼承自View,SurfaceView或其他的View。

三.幾個重要的函式

建構函式

建構函式是View的入口,可以用於初始化一些的內容,和獲取自定義屬性。
View的建構函式有四種過載分別如下:


    public CustomView(Context context)
    {
        super
(context); } public CustomView(Context context, AttributeSet attrs) { super(context, attrs); } public CustomView(Context context, AttributeSet attrs, int defStyleAttr) { super(context, attrs, defStyleAttr); } public CustomView(Context context, AttributeSet attrs, int
defStyleAttr, int defStyleRes) { super(context, attrs, defStyleAttr, defStyleRes); }

有四個引數的建構函式在API21的時候才新增上,暫不考慮。

有三個引數的建構函式中第三個引數是預設的Style,這裡的預設的Style是指它在當前Application或Activity所用的Theme中的預設Style,且只有在明確呼叫的時候才會生效,以系統中的ImageButton為例說明:

public ImageButton(Context context, AttributeSet attrs) {
    //呼叫了三個引數的建構函式,明確指定第三個引數
this(context, attrs, com.android.internal.R.attr.imageButtonStyle); } public ImageButton(Context context, AttributeSet attrs, int defStyleAttr) { //此處調了四個引數的建構函式,無視即可 this(context, attrs, defStyleAttr, 0); }

注意:即使你在View中使用了Style這個屬性也不會呼叫三個引數的建構函式,所呼叫的依舊是兩個引數的建構函式。

由於三個引數的建構函式第三個引數一般不用,暫不考慮,第三個引數的具體用法會在以後用到的時候詳細介紹。

排除了兩個之後,只剩下一個引數和兩個引數的建構函式,他們的詳情如下:

//一般在直接New一個View的時候呼叫。
public void CustomView(Context context) {}

//一般在layout檔案中使用的時候會呼叫,關於它的所有屬性(包括自定義屬性)都會包含在attrs中傳遞進來。
public void CustomView(Context context, AttributeSet attrs) {}

以下方法呼叫的是一個引數的建構函式:

//在Activity中
CustomView customView = new CustomView(this);

以下方法呼叫的是兩個引數的建構函式:

 <com.zhoujian.viewevent.view.CustomView
       android:layout_width="wrap_content"
       android:layout_height="wrap_content"/>

測量View大小(onMeasure)

為什麼要測量View大小?

View的大小不僅由自身所決定,同時也會受到父控制元件的影響,為了我們的控制元件能更好的適應各種情況,一般會自己進行測量。

測量View大小使用的是onMeasure函式,我們可以從onMeasure的兩個引數中取出寬高的相關資料:

@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
    int widthsize  MeasureSpec.getSize(widthMeasureSpec);      //取出寬度的確切數值
    int widthmode  MeasureSpec.getMode(widthMeasureSpec);      //取出寬度的測量模式

    int heightsize  MeasureSpec.getSize(heightMeasureSpec);    //取出高度的確切數值
    int heightmode  MeasureSpec.getMode(heightMeasureSpec);    //取出高度的測量模式
}

從上面可以看出 onMeasure 函式中有 widthMeasureSpec 和 heightMeasureSpec 這兩個 int 型別的引數, 毫無疑問他們是和寬高相關的, 但它們其實不是寬和高, 而是由寬、高和各自方向上對應的測量模式來合成的一個值:

測量模式一共有三種, 被定義在 Android 中的 View 類的一個內部類View.MeasureSpec中:

模式 描述
UNSPECIFIED 預設值,父控制元件沒有給子view任何限制,子View可以設定為任意大小
EXACTLY 表示父控制元件已經確切的指定了子View的大小
AT_MOST 表示子View具體大小沒有尺寸限制,但是存在上限,上限一般為父View大小

注意: 用 MeasureSpec 的 getSize是獲取數值, getMode是獲取模式。如果對View的寬高進行修改了,不要呼叫 super.onMeasure( widthMeasureSpec, heightMeasureSpec); 要呼叫 setMeasuredDimension( widthsize, heightsize); 這個函式。

確定View大小(onSizeChanged)

這個函式在檢視大小發生改變時呼叫。

在測量完View並使用setMeasuredDimension函式之後View的大小基本上已經確定了,那麼為什麼還要再次確定View的大小呢?

這是因為View的大小不僅由View本身控制,而且受父控制元件的影響,所以我們在確定View大小的時候最好使用系統提供的onSizeChanged回撥函式。

@Override
protected void onSizeChanged(int w, int h, int oldw, int oldh) {
    super.onSizeChanged(w, h, oldw, oldh);
}

可以看出,它又四個引數,分別為 寬度,高度,上一次寬度,上一次高度。

這個函式比較簡單,我們只需關注 寬度(w), 高度(h) 即可,這兩個引數就是View最終的大小。

確定子View佈局位置(onLayout)

確定佈局的函式是onLayout,它用於確定子View的位置,在自定義ViewGroup中會用到,他呼叫的是子View的layout函式。

在自定義ViewGroup中,onLayout一般是迴圈取出子View,然後經過計算得出各個子View位置的座標值,然後用以下函式設定子View位置。

child.layout(l, t, r, b);

四個引數分別為:

名稱 說明 對應的函式
l View左側距父View左側的距離 getLeft();
t View頂部距父View頂部的距離 getTop();
r View右側距父View左側的距離 getRight();
b View底部距父View頂部的距離 getBottom();

繪製內容(onDraw)

onDraw是實際繪製的部分,也就是我們真正關心的部分,使用的是Canvas繪圖。

@Override
protected void onDraw(Canvas canvas) {
    super.onDraw(canvas);
}

完整的程式碼

package com.zhoujian.viewevent.view;

import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Color;
import android.util.AttributeSet;
import android.view.View;

/**
 * Created by zhoujian on 2017/1/19.
 */

public class CustomView extends View {


    public CustomView(Context context)
    {
        super(context);
    }

    /**
     * 初始化一些的內容,和獲取自定義屬性。
     * @param context
     * @param attrs
     */

    public CustomView(Context context, AttributeSet attrs)
    {
        super(context, attrs);
    }

    public CustomView(Context context, AttributeSet attrs, int defStyleAttr)
    {
        super(context, attrs, defStyleAttr);
    }


    /**
     * 測量view大小
     * @param widthMeasureSpec
     * @param heightMeasureSpec
     */
    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec)
    {
        int widthsize = MeasureSpec.getSize(widthMeasureSpec);      //取出寬度的確切數值
        int widthmode = MeasureSpec.getMode(widthMeasureSpec);      //取出寬度的測量模式
        int heightsize = MeasureSpec.getSize(heightMeasureSpec);    //取出高度的確切數值
        int heightmode = MeasureSpec.getMode(heightMeasureSpec);    //取出高度的測量模式
        setMeasuredDimension(widthsize,heightsize);
    }

    /**
     * 確定view大小
     * @param w
     * @param h
     * @param oldw
     * @param oldh
     */

    @Override
    protected void onSizeChanged(int w, int h, int oldw, int oldh)
    {
        super.onSizeChanged(w, h, oldw, oldh);
    }

    /**
     * 確定子View佈局位置
     * @param changed
     * @param left
     * @param top
     * @param right
     * @param bottom
     */

    @Override
    protected void onLayout(boolean changed, int left, int top, int right, int bottom)
    {
        super.onLayout(changed, left, top, right, bottom);
    }

    /**
     * 繪製內容
     * @param canvas
     */

    @Override
    protected void onDraw(Canvas canvas)
    {
        canvas.drawColor(Color.RED); //繪製藍色
    }
}

對外提供操作方法和監聽回撥

自定義完View之後,一般會對外暴露一些介面,用於控制View的狀態等,或者監聽View的變化.

參考文章

相關推薦

Android定義View高階()-分類流程

一.自定義View繪製流程 二.自定義View分類 自定義ViewGroup 自定義ViewGroup一般是利用現有的元件根據特定的佈局方式來組成新的元件,大多繼承自ViewGroup或各種Layout。 自定義View 在沒有現成的View,

安卓定義View進階-分類流程

本章節為什麼要叫進階篇?(雖然講的是基礎內容),因為從本篇開始,將會逐漸揭開自定義View的神祕面紗,每一篇都將比上一篇內容更加深入,利用所學的知識能夠製作更加炫酷自定義View,就像在臺階上一樣,每一篇都更上一層,幫助大家一步步走向人生巔峰,出任CEO,迎娶白富美。 誤

Android 定義View

前言:可是有時候我們總感覺官方定義的一些基本元件不夠用,自定義元件就不可避免了。那麼如何才能做到像官方提供的那些元件一樣用xml來定義他的屬性呢? 先總結下自定義View的步驟: 1、自定義View的屬性; 2、在View的構造方法中獲得自定義的屬性。 一、在re

android定義View)、正弦波水波紋

文章目錄 1、正弦曲線知識 2、靜態正弦曲線繪製 3、動態正弦曲線繪製 4、完整原始碼 1、正弦曲線知識 對這個初中知識遺忘了的可以先看看正弦曲線百度百科詞條方便加深理解。

Android定義view),打造絢麗的驗證碼

前言:我相信信念的力量,信念可以支撐起一個人,一個名族,一個國家。正如“人沒有夢想和鹹魚有什麼區別”一樣,我有信念,有理想,所以我正在努力向著夢想前進~。 自定義view,如果是我,我首先要看到自定義view的效果圖,然後再想想怎麼實現這種效果或功能,所以先貼

Android定義view案例氣泡框

通過之前的兩個基礎文章的學習,我們基本知道自定義view的基本流程和各方法的作用了。那麼接下來我們就要拿起畫筆去繪製我們的view了。首先老規矩把我們的學習大綱拿出來,時刻不能忘,,哈哈、 1.自定義view單純的用畫筆繪製view(死view) 2.自定義view增加手勢

Android 定義View高階特效,神奇的貝塞爾曲線

初始化引數 private static final String TAG = "BIZIER"; private static final int LINEWIDTH = 5; private static final int POINTWIDTH = 10; private Context mContex

Android定義View總結()基礎知識例項

自定義View是最能體現一個Android開發者水平的技能之一了。 接下來的一些列部落格將總結一下Android的自定義相關View知識,包括View的結構,事件體系,工作原理,自定義View的繪製等。 參考資料部分來自於書上以及各種部落格。 新建了一個qq群 482

Android定義View進階:分類流程

經歷過前面三篇囉囉嗦嗦的基礎篇之後,終於到了進階篇,正式進入解析自定義View的階段。 前言 本章節為什麼要叫進階篇?(雖然講的是基礎內容),因為從本篇開始,將會逐漸揭開自定義View的神祕面紗,每一篇都將比上一篇內容更加深入,利用所學的知識能夠製作更加炫酷自定義View,就像

Android 定義View——分類流程

1.View 分類與流程 1.1View 分類 1.自定義ViewGroup 自定義ViewGroup一般是利用現有的元件根據特定的佈局方式來組成新的元件,大多繼承自ViewGroup或各種Layout,包含有子View。 ViewGroup 及它

【朝花夕拾】Android定義View篇之(十View的滑動,彈性滑動定義PagerView

前言        由於手機螢幕尺寸有限,但是又經常需要在螢幕中顯示大量的內容,這就使得必須有部分內容顯示,部分內容隱藏。這就需要用一個Android中很重要的概念——滑動。滑動,顧名思義就是view從一個地方移動到另外一個地方,我們平時看到的

Android定義viewactivity的傳值

重復 轉動 自定義 activit 廣播 內部 代碼 view 等待 昨晚在寫團隊項目的時候,遇到一個問題,直到今天早上才解決。。。即在自定義view“轉盤”結束轉動後獲取結果的處理中,我是想吧值傳到activity中的一個textview中的,但我的自定義view類不是a

Android定義View--翻書控制元件(

0.前言 最近重看了一遍封神演義,感覺QQ閱讀那個翻書的效果挺好的,準備做一個。上週五下午用了兩個小時只寫了一部分功能,以後有時間再完善 1.分析 先看效果圖 這個空間,說簡單也簡單,說難也難,簡單就在於這個效果主要就是依賴canvas的clippath才見到部分canvas,難就難在裁

Android學習—簡單定義View

最近手上不忙所以回顧了一下自己今年來所接觸和學習的東西,突然覺得寫部落格真是一個很好的方式,希望自己 可以堅持下去。 自定義View的流程 建立自定義類繼承View,並重寫構造方法,構造方法總共有四種,我們暫時只需要繼承前兩種 public CircleVi

Android定義View的實現方法,帶你步步深入瞭解View(四)

不知不覺中,帶你一步步深入瞭解View系列的文章已經寫到第四篇了,回顧一下,我們一共學習了LayoutInflater的原理分析、檢視的繪製流程、檢視的狀態及重繪等知識,算是把View中很多重要的知識點都涉及到了。如果你還沒有看過我前面的幾篇文章,建議先去閱讀一下,多瞭解一些

Android定義View的實現方法 帶你步步深入瞭解View

                不知不覺中,帶你一步步深入瞭解View系列的文章已經寫到第四篇了,回顧一下,我們一共學習了LayoutInflater的原理分析、檢視的繪製流程、檢視的狀態及重繪等知識,算是把View中很多重要的知識點都涉及到了。如果你還沒有看過我前面的幾篇文章,建議先去閱讀一下,多瞭解一些原

Android定義View教你步實現即刻點贊效果

前言 今天朋友看了HenCoder的自定義View後說,HenCoder對自定義View講的不錯。實踐中仿寫即刻的點贊你有思路嗎,你不實現一下?二話不說,看了朋友手機效果,對他說:實現不難,用到了位移,縮放,漸變動畫和自定義View的基礎用法,好,那我實現一下,剛好加深對自定義View的理解。 素材準備

Android——定義View(學習Android開發藝術探索)

ViewRoot和DecorView ViewRoot對應於ViewRootImpl類,是連線WindowManager和DecorView的紐帶,View的三大流程均是通過ViewRoot來完成的。在ActivityThread中,當Activity物件被建

Android 定義View ()

轉載請標明出處:http://blog.csdn.net/lmj623565791/article/details/24252901 很多的Android入門程式猿來說對於Android自定義View,可能都是比較恐懼的,但是這又是高手進階的必經之路,所有準備在自定義Vie

Android定義View

為什麼要自定義View android提供了很多控制元件供我們使用 但有些功能是系統所提供的實現不 了的 這時候我們就需要自定義一個View來實現我們所需要的效果. 在Android中所有的控制元件都直接或間接的繼承自View,分View和ViewGr