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(); } }); } }
運行結果:
原創文章,轉載請註明出處。