Android中自定義組合控件

Android中自定義控件的狀況很是多,通常自定義控件能夠分爲兩種:繼承控件及組合控件。前者是經過繼承View或其子類,重寫方法實現自定義的顯示及事件處理方式;後者是經過組合已有的控件,來實現結構的簡化和代碼的重用。java

本篇文章主要介紹自定義組合控件,繼承控件後續有機會再述。android

自定義組合控件通常來講都是以ViewGroup及其子類(LinearLayout、RelativeLayout、FrameLayout等)爲主,內部嵌套其餘控件,來組合成一個新的控件,實現一些特定的須要,能夠是代碼簡化,結構清晰,重用性較高。app

一般來講,咱們會實現定義好一個Layout.xml文件,而後讓咱們的自定義控件去加載此xml,並獲取子控件,而後設置屬性(能夠經過代碼,也能夠從資源文件中加載)、添加事件。eclipse

自定義要點:ide

1.加載xml文件是在構造方法中完成的,經過調用inflate(R.layout.my_layout,this,true),注意第二個和第三個參數;工具

2.若是須要從資源文件中加載自定義的屬性,則必須重寫Constructor(Context context, AttributeSet attrs)此構造方法,屬性是定義在attrs.xml中的;佈局

3.獲取子控件對象,能夠在構造方法中獲取,也能夠重寫onFinishInflate()方法來獲取,我的建議採用第二種,能夠保證控件已經徹底加載好了;開發工具

4.添加事件能夠直接在控件中寫,不過考慮到擴展性及複用性,建議對外暴露接口this

示例代碼(代碼比較簡單,只是描述一下思路)spa

自定義控件layout:header.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="wrap_content">
    <ImageButton android:id="@+id/ib_header"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_alignParentRight="true"
        android:layout_centerVertical="true"
        android:src="@android:drawable/ic_menu_zoom" />
    <TextView android:id="@+id/tv_header"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_centerInParent="true" />

</RelativeLayout>

自定義控件類:Header.java

package com.ivan.app1.widgets;

import com.ivan.app1.R;

import android.content.Context;
import android.content.res.TypedArray;
import android.graphics.Color;
import android.text.TextUtils;
import android.util.AttributeSet;
import android.view.LayoutInflater;
import android.widget.ImageButton;
import android.widget.LinearLayout;
import android.widget.TextView;

/**
 * 自定義標題欄組合控件,內部包含一個TextView和一個ImageButton
 * User: xyh
 * Date: 2015/6/2
 * Time: 9:39
 */
public class Header extends RelativeLayout {

    private TextView mTextView;
    private ImageButton mImageButton;

    private String titleText;
    private int titleTextColor;
    private float titleTextSize;

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

    public Header(Context context, AttributeSet attrs) {
        super(context, attrs);

        //加載視圖的佈局
        LayoutInflater.from(context).inflate(R.layout.header,this,true);

        //加載自定義的屬性
        TypedArray a=context.obtainStyledAttributes(attrs,R.styleable.Header);
        titleText=a.getString(R.styleable.Header_titleText);
        titleTextColor=a.getColor(R.styleable.Header_titleTextColor, Color.WHITE);
        titleTextSize=a.getDimension(R.styleable.Header_titleTextSize,20f);

        //回收資源,這一句必須調用
        a.recycle();
    }

    /**
     * 此方法會在全部的控件都從xml文件中加載完成後調用
     */
    @Override
    protected void onFinishInflate() {
        super.onFinishInflate();

        //獲取子控件
        mTextView= (TextView) findViewById(R.id.tv_header);
        mImageButton= (ImageButton) findViewById(R.id.ib_header);

        //將從資源文件中加載的屬性設置給子控件
        if (!TextUtils.isEmpty(titleText))
            setPageTitleText(titleText);
        setPageTitleTextColor(titleTextColor);
        setPageTitleTextSize(titleTextSize);

    }

    /**
     * 設置標題文字
     * @param text
     */
    public void setPageTitleText(String text) {
        mTextView.setText(text);
    }

    /**
     * 設置標題文字顏色
     * @param color
     */
    public void setPageTitleTextColor(int color) {
        mTextView.setTextColor(color);
    }

    /**
     * 設置標題文字大小
     * @param size
     */
    public void setPageTitleTextSize(float size) {
        mTextView.setTextSize(size);
    }

    /**
     * 設置按鈕點擊事件監聽器
     * @param listener
     */
    public void setOnHeaderClickListener(OnClickListener listener) {
        mImageButton.setOnClickListener(listener);
    }
}

自定義屬性文件:attrs.xml

<?xml version="1.0" encoding="utf-8"?>
<resources>
    <!-- 自定義的屬性-->
    <declare-styleable name="Header">
        <attr name="titleTextSize" format="dimension" />
        <attr name="titleTextColor" format="color" />
        <attr name="titleText" format="string"/>
    </declare-styleable>
</resources>

如下是引用方式,activity佈局文件:main.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:orientation="vertical"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

    <!-- 注意須要加上命名空間 在eclipse開發工具中:使用 xmlns:app="http://schemas.android.com/apk/res/com.ivan.app1.widgets"
         在IntelliJ Idea或者Android Studio中以Gradle構建時,使用 xmlns:app="http://schemas.android.com/apk/res-auto"
    -->

    <!-- 經過包的類的全名來引用自定義視圖-->
    <com.ivan.app1.widgets.Header
        android:id="@+id/header"
        android:layout_width="match_parent"
        android:layout_height="48dp"
        android:background="@color/black"
        app:titleText="我是標題"
        app:titleTextColor="#ff0000"
        app:titleTextSize="12sp"/>

    <TextView android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:gravity="center"
        android:text="我是內容"
        android:textSize="60sp"/>
</LinearLayout>

 

 主Activity類:MainActivity.java

package com.ivan.app1;

import com.ivan.app1.widgets.Header;

import android.os.Bundle;
import android.support.v7.app.AppCompatActivity;
import android.view.View;
import android.widget.Toast;

/**
 * User: xyh
 * Date: 2015/6/2
 * Time: 10:30
 */
public class MainActivity extends AppCompatActivity {

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);

        ((Header)findViewById(R.id.header)).setOnHeaderClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                Toast.makeText(getApplicationContext(),"標題欄的按鈕被點擊了",Toast.LENGTH_LONG).show();
            }
        });
    }
}

運行結果:

原創文章,轉載請註明出處。

相關文章
相關標籤/搜索