Android自訂表格格控制項滿足人們對視覺的需求

來源:互聯網
上載者:User

Android平台已經給我們提供了很多標準的組件,如:TextView、EditView、Button、ImageView、Menu等,還有許多布局控制項,常見的有:AbsoluteLayout、LinerLayout、RelativeLayout、TableLayout等。但隨著人們對視覺的需求,基本組件已無法滿足人們求新求異的要求,於是我們常常會自訂群組件,用來實現更美觀的UI介面。

實現自訂控制項通常有兩種途徑,一種是繼承View類,重寫其中的重要方法,另一種是繼承ViewGroup類,通過重寫父類中的有些方法,達到重新繪製組件的目的。最近做了一個自訂表格格控制項的練習,從中總結到一些經驗。在這個練習中,我通過繼承ViewGroup類,重新繪製了用於呈現表格樣式的容器組件,首先來看一下父類ViewGroup。該類有三個構造方法:ViewGroup(Context context)、ViewGroup(Context context,AttributeSet attrs)、ViewGroup(Context context,AttributeSet attrs,int defStyle),我們自訂的繼承ViewGroup的類需要實現它的至少一個構造方法。ViewGroup中有幾個方法非常重要,這幾個方法更好的協助我們實現自己的組件的布局與繪製。

1、onLayout方法

該方法用於在容器中如何擺放子控制項,如果不重寫該方法,子控制項將無法在布局控制項中得以展示,該方法有五個參數,用於設定子控制項的上下左右四個邊框的位置,還有一個標誌位,這個方法也是子類必須實現的,因為該方法是個抽象方法。

2、addView方法

該方法用於在容器組件中添加子控制項

3、dispatchDraw方法

通過該方法,我們可以擷取canvas對象,該對象允許我們在組件上畫任意我們想要的圖形,在這個表格控制項中,我們可以在畫布上上表格的外邊框及表格線

4、getChildCount和getChildAt方法

這兩個方法用於擷取該容器控制項中子控制項的數目和位置,便於我們對子控制項的排版和布局

5、onMeasure方法

這個方法是用來測量子控制項大小的,它在onLayout方法之前被調用,測量了子控制項的大小尺寸,然後可以繪製子控制項在容器組件中的布局位置

下面直接給出程式碼範例,僅供參考

首先是表格控制項的類:

複製代碼 代碼如下:public class TableView extends ViewGroup{
private static final int STARTX = 0;// 起始X座標
private static final int STARTY = 0;// 起始Y座標
private static final int BORDER = 2;// 表格邊框寬度
private int mRow;// 行數
private int mCol;// 列數
public TableView(Context context, AttributeSet attrs) {
super(context, attrs);
this.mRow = 3;// 預設行數為3
this.mCol = 3;// 預設列數為3
// 添加子控制項
this.addOtherView(context);
}
public TableView(Context context, int row,int col) {
super(context);
if(row>20 || col>20){
this.mRow = 20;// 大於20行時,設定行數為20行
this.mCol = 20;// 大於20列時,設定列數為20列
}else if(row==0 || col==0){
this.mRow = 3;
this.mCol = 3;
}
else{
this.mRow = row;
this.mCol = col;
}
// 添加子控制項
this.addOtherView(context);
}
public void addOtherView(Context context){
int value = 1;
for(int i=1;i<=mRow;i++){
for(int j=1;j<=mCol;j++){
TextView view = new TextView(context);
view.setText(String.valueOf(value++));
view.setTextColor(Color.rgb(79, 129, 189));
view.setGravity(Gravity.CENTER);
if(i%2==0){
view.setBackgroundColor(Color.rgb(219, 238, 243));
}else{
view.setBackgroundColor(Color.rgb(235, 241, 221));
}
this.addView(view);
}
}
}
@Override
protected void dispatchDraw(Canvas canvas) {
Paint paint = new Paint();
paint.setStrokeWidth(BORDER);
paint.setColor(Color.rgb(79, 129, 189));
paint.setStyle(Style.STROKE);
// 繪製外部邊框
canvas.drawRect(STARTX, STARTY, getWidth()-STARTX, getHeight()-STARTY, paint);
// 畫列分割線
for(int i=1;i<mCol;i++){
canvas.drawLine((getWidth()/mCol)*i, STARTY, (getWidth()/mCol)*i, getHeight()-STARTY, paint);
}
// 畫行分割線
for(int j=1;j<mRow;j++){
canvas.drawLine(STARTX, (getHeight()/mRow)*j, getWidth()-STARTX, (getHeight()/mRow)*j, paint);
}
super.dispatchDraw(canvas);
}
@Override
protected void onLayout(boolean changed, int l, int t, int r, int b) {
int x = STARTX+BORDER;
int y = STARTY+BORDER;
int i = 0;
int count = getChildCount();
for(int j=0; j<count; j++){
View child = getChildAt(j);
child.layout(x, y, x+getWidth()/mCol-BORDER*2, y+getHeight()/mRow-BORDER*2);
if(i >=(mCol-1)){
i = 0;
x = STARTX+BORDER;
y += getHeight()/mRow;
}else{
i++;
x += getWidth()/mCol;
}
}
}
public void setRow(int row){
this.mRow = row;
}
public void setCol(int col){
this.mCol = col;
}
}

然後我們在Activity中使用我們的控制項:複製代碼 代碼如下:public class MainActivity extends Activity implements OnClickListener{
private Button btn;
private EditText row;
private EditText col;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
btn = (Button)findViewById(R.id.button1);
row = (EditText)findViewById(R.id.editRow);
col = (EditText)findViewById(R.id.editCol);
row.setError("請輸入小於20的整數");
col.setError("請輸入小於20的整數");
btn.setOnClickListener(this);
}
@Override
public boolean onCreateOptionsMenu(Menu menu) {
// Inflate the menu; this adds items to the action bar if it is present.
getMenuInflater().inflate(R.menu.main, menu);
return true;
}
@Override
public void onClick(View v) {
Intent intent = new Intent();
Bundle bun = new Bundle();
if("".equals(row.getText().toString())){
Toast.makeText(this, "行數不可為空", Toast.LENGTH_SHORT).show();
return;
}else if("".equals(col.getText().toString())){
Toast.makeText(this, "列數不可為空", Toast.LENGTH_SHORT).show();
return;
}else{
int rowNum = Integer.parseInt(row.getText().toString());
int colNum = Integer.parseInt(col.getText().toString());
bun.putInt("row", rowNum);
bun.putInt("col", colNum);
intent.setClass(MainActivity.this, TableActivity.class);
intent.putExtras(bun);
startActivity(intent);
}
}
}

複製代碼 代碼如下:public class TableActivity extends Activity{
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
Intent intent = this.getIntent();
Bundle bun = intent.getExtras();
int row = bun.getInt("row");
int col = bun.getInt("col");
TableView table = new TableView(this, row, col);
setContentView(table);
}
}

如下:

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.