版權聲明:本文為部落客原創文章,轉載請注明出處http://blog.csdn.net/u013132758。 https://blog.csdn.net/u013132758/article/details/50096959
View元件的作用類似于SWing程式設計中的JPanel,它隻是一個矩形的空白區域,View元件沒有任何内容。對于Android應用的其他UI元件來說,它們都繼承了View元件,然後在View元件提供的空白區域上繪制外觀。
基于Android UI元件的實作原理,開發者完全可以開發出項目定制的元件——當Android系統提供的Ui元件不足以滿足項目需求時,我們可以通過繼承自View來自定義View元件。
當我們打算派生自己的UI元件時,首先定義一個繼承自View的子類,然後重寫View類的一個或多個方法。
1、通常可以被使用者重寫的方法如下:
構造器:重寫構造器是定制View最基本的方式,當Java代碼建立一個View執行個體,或者根據XML布局檔案加載并建構界面時将需要該構造器。
onFinishInflate():這是一個回調方法,當應用從XML布局檔案加載該元件并利用它來建構界面之後,該方法将被回調。
onMeasure(int,int):調用該方法來檢測View元件及其包含的所有子元件的大小。
onLayout(boolean,int,int,int,int):當該元件需要重新配置設定位置、大小時,該方法被回調。
onSizeChanged(int,int,int,int):當該元件大小被改變時回調該方法。
onDraw(Canvas):當該元件需要繪制它的内容時,回調該方法進行繪制。
onKeyDown(int,KeyEvent):當某個鍵被按下時回調該方法。
onKeyUp(int,KeyEvent):當松開摸個按鍵時回調該方法。
onTrackballEvent(MotionEvent):當發生軌迹球事件時觸發該方法。
onTouchEvent(MotionEvent):當發生觸摸屏事件時觸發該方法。
onFocusChanged(boolean gainFocus,int direction,Rect previouslyFocuseRect):當該元件焦點發生改變時觸發該方法。
onWindoeFocusChanged(boolean):當包含該元件的視窗失去或獲得焦點時觸發該方法。
onAttachedToWondow():當把該元件放入某個視窗時觸發該方法。
onDetachedFromWindow():當把該元件從某個視窗分離時觸發該方法。
onWindowVisibilityChanged(int):當包含該元件的視窗的可見性發生改變時觸發該方法。
當我們開發自己的元件時并不一定要重寫全部,根據需要寫就可以。
2、執行個體:跟手指的小球
下面的DrawView元件繼承自View基類,并重寫了onDraw方法和onTouchEvent方法。
package com.example.leiqi.myview;
import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.util.AttributeSet;
import android.view.MotionEvent;
import android.view.View;
/**
* Created by leiqi on 15-11-29.
*/
public class DrawView extends View {
public float currentX = 40;
public float currentY = 50;
// 定義并建立畫筆
Paint p = new Paint();
public DrawView(Context context)
{
super(context);
}
private DrawView(Context context,AttributeSet set)
{
super(context,set);
}
@Override
public void onDraw(Canvas canvas){
super.onDraw(canvas);
//設定畫筆的顔色
p.setColor(Color.BLUE);
//繪制一個小圓(作為小球)
canvas.drawCircle(currentX, currentY, 15, p);
}
//為該元件觸碰事件寫處理方法
@Override
public boolean onTouchEvent(MotionEvent event)
{
//修改currentX,currentY兩個參數
currentX = event.getX();
currentY = event.getY();
//通知目前元件重繪
invalidate();
//傳回true,表明處理方法已正确處理該事件
return true;
}
}
有了這個自定義元件後接下來就是在java代碼中将元件添加到指定容器中。
package com.example.leiqi.myview;
import android.os.Bundle;
import android.support.design.widget.FloatingActionButton;
import android.support.design.widget.Snackbar;
import android.support.v7.app.AppCompatActivity;
import android.support.v7.widget.Toolbar;
import android.view.View;
import android.view.Menu;
import android.view.MenuItem;
import android.widget.RelativeLayout;
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
RelativeLayout root = (RelativeLayout) findViewById(R.id.root);
//建立DrawerView元件
final DrawView drawView = new DrawView(this);
drawView.setMinimumHeight(500);
drawView.setMinimumWidth(300);
root.addView(drawView);
}
}
布局檔案:
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:id="@+id/root"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:showIn="@layout/activity_main">
<TextView
android:gravity="center"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Hello World!" />
</RelativeLayout>
我們也可以直接在布局檔案中添加自定義元件
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:id="@+id/root"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:showIn="@layout/activity_main">
<com.example.leiqi.myview.DrawView
android:layout_width="match_parent"
android:layout_height="wrap_content" />
<TextView
android:gravity="center"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Hello World!" />
</RelativeLayout>
上面布局檔案中加了元件那麼我們隻需要在java檔案中加載出組建就好。
package com.example.leiqi.myview;
import android.os.Bundle;
import android.support.v7.app.AppCompatActivity;
import android.widget.RelativeLayout;
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
}