⑴ android LinearLayout 裡面的東西怎麼換行
由於前段時間項目中使用到了自動換行的線性布局,本來打算用表格布局在裡面一個個的用java代碼添加ImageView的,但是添加的View控制項是不確定的,因為得靠伺服器的數據返回,就這樣手動用Java代碼畫布局的方式就這樣夭折了,因為在表哥布局中我無法確定一行顯示多少個ImageView的數目,所以無法動態添加,最後自能自己去看看那種能夠換行的線性布局了,線性布局比較不好的是不能自動換行,也就是當設置LinearLayout的orentation 設置為vertical 為豎直方向也就是只有一列,每行只能顯示一個View或者View的子類,當設置LinearLayout的orentitation為Horizontal,LinearLayout的只能顯示為一行,橫向顯示,當屏幕滿了的時候,View控制項並不會自動換行,所以我們要做的就是在LinearLayout滿的時候自動換行。
需要了解的是怎麼樣繪制根據子控制項的長寬繪制父控制項的寬度與高度,所以需要傳入的參數控制項的高度,視圖分為兩種一種是View類型的,代表控制項有TextView,Button,EditText 等等,還有一種是裝視圖的容器控制項繼承自ViewGroup的控制項,如LinearLayout,RelativeLayout,TabHost等等控制項,需要自動換行的線性布局的話,就需要根據子控制項的高度與寬度,來動態載入父控制項的高度與寬度,所以需要在構造函數中傳入每一個子控制項的固定的高度,或者是動態設置子控制項的高度與寬度。
將自定義的LinearLayout 也繼承自ViewGroup 並且重寫抽象類ViewGrouop的幾個方法:onMeasure(),onLayout(),dispathDraw() 三個方法的意思分別是:第一個onMeasure()是用來計算控制項以及子控制項所佔用的區域,第二個onLayout()是控制子控制項的換行,第三個可寫可不寫,主要是用來繪制控制項的邊框,
自定義LinearLayout的代碼如下:
[java] view plainprint?
package com.huanglong.mylinearlayout;
import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.graphics.Rect;
import android.util.AttributeSet;
import android.view.View;
import android.view.ViewGroup;
/**
* @author huanglong 2013-5-28 自定義自動換行LinearLayout
*/
public class FixGridLayout extends ViewGroup {
private int mCellWidth;
private int mCellHeight;
public FixGridLayout(Context context) {
super(context);
}
public FixGridLayout(Context context, AttributeSet attrs) {
super(context, attrs);
}
public FixGridLayout(Context context, AttributeSet attrs, int defStyle) {
super(context, attrs, defStyle);
}
public void setmCellWidth(int w) {
mCellWidth = w;
requestLayout();
}
public void setmCellHeight(int h) {
mCellHeight = h;
requestLayout();
}
/**
* 控制子控制項的換行
*/
@Override
protected void onLayout(boolean changed, int l, int t, int r, int b) {
int cellWidth = mCellWidth;
int cellHeight = mCellHeight;
int columns = (r - l) / cellWidth;
if (columns < 0) {
columns = 1;
}
int x = 0;
int y = 0;
int i = 0;
int count = getChildCount();
for (int j = 0; j < count; j++) {
final View childView = getChildAt(j);
// 獲取子控制項Child的寬高
int w = childView.getMeasuredWidth();
int h = childView.getMeasuredHeight();
// 計運算元控制項的頂點坐標
int left = x + ((cellWidth - w) / 2);
int top = y + ((cellHeight - h) / 2);
// int left = x;
// int top = y;
// 布局子控制項
childView.layout(left, top, left + w, top + h);
if (i >= (columns - 1)) {
i = 0;
x = 0;
y += cellHeight;
} else {
i++;
x += cellWidth;
}
}
}
/**
* 計算控制項及子控制項所佔區域
*/
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
// 創建測量參數
int cellWidthSpec = MeasureSpec.makeMeasureSpec(mCellWidth, MeasureSpec.AT_MOST);
int cellHeightSpec = MeasureSpec.makeMeasureSpec(mCellHeight, MeasureSpec.AT_MOST);
// 記錄ViewGroup中Child的總個數
int count = getChildCount();
// 設置子空間Child的寬高
for (int i = 0; i < count; i++) {
View childView = getChildAt(i);
/*
* 090 This is called to find out how big a view should be. 091 The
* parent supplies constraint information in the width and height
* parameters. 092 The actual mesurement work of a view is performed
* in onMeasure(int, int), 093 called by this method. 094 Therefore,
* only onMeasure(int, int) can and must be overriden by subclasses.
* 095
*/
childView.measure(cellWidthSpec, cellHeightSpec);
}
// 設置容器控制項所佔區域大小
// 注意setMeasuredDimension和resolveSize的用法
setMeasuredDimension(resolveSize(mCellWidth * count, widthMeasureSpec),
resolveSize(mCellHeight * count, heightMeasureSpec));
// setMeasuredDimension(widthMeasureSpec, heightMeasureSpec);
// 不需要調用父類的方法
// super.onMeasure(widthMeasureSpec, heightMeasureSpec);
}
/**
* 為控制項添加邊框
*/
@Override
protected void dispatchDraw(Canvas canvas) {
// 獲取布局控制項寬高
int width = getWidth();
int height = getHeight();
// 創建畫筆
Paint mPaint = new Paint();
// 設置畫筆的各個屬性
mPaint.setColor(Color.BLUE);
mPaint.setStyle(Paint.Style.STROKE);
mPaint.setStrokeWidth(10);
mPaint.setAntiAlias(true);
// 創建矩形框
Rect mRect = new Rect(0, 0, width, height);
// 繪制邊框
canvas.drawRect(mRect, mPaint);
// 最後必須調用父類的方法
super.dispatchDraw(canvas);
}
}
然後在Xml文件中引用自己定義的控制項,在Java代碼中調用:
[java] view plainprint?
package com.huanglong.mylinearlayout;
import android.os.Bundle;
import android.app.Activity;
import android.view.Menu;
import android.view.MenuItem;
import android.widget.CheckBox;
import android.widget.SimpleAdapter;
import android.support.v4.app.NavUtils;
public class MainActivity extends Activity {
private SimpleAdapter adapter;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
FixGridLayout fixGridLayout = (FixGridLayout) findViewById(R.id.ll);
fixGridLayout.setmCellHeight(30);
fixGridLayout.setmCellWidth(100);
for (int i = 0; i < 7; i++) {
CheckBox box = new CheckBox(MainActivity.this);
box.setText("第"+i+"個");
fixGridLayout.addView(box);
}
}
@Override
public boolean onCreateOptionsMenu(Menu menu) {
getMenuInflater().inflate(R.menu.activity_main, menu);
return true;
}
}
⑵ android開發問題: 在Activity主界面繪制畫布,可是它把整個窗口都覆蓋掉了,怎麼設定畫布的窗口大小呢(
你是說View占滿了整個屏幕嗎?這樣的話就只能在layout中寫一個布局文件,將View以一種組件的方式聲明到.xml文件中,然後你可以在它的上下中聲明其他的button或者textview或者ImageButton等。
⑶ Android 自定義View中onMeasure中使用resolveSize是什麼意思
onmeasure方法主要是測量控制項本身的大小(width,height)以及它的子控制項的大小
不太明白你說的畫布大小,但ondraw需要畫的所有東西的大小,都要在onmeasure中測量獲取
⑷ Android問題:View.resolveSizeAndState()方法的第三個參數什麼意思方法返回值是什麼意思
public
static
final
int
MEASURED_HEIGHT_STATE_SHIFT
Added in API level 11
Bit shift of MEASURED_STATE_MASK to get to the height bits
for functions that combine both width and height into a single int,
such as getMeasuredState() and the childState argument of
resolveSizeAndState(int, int, int).
Constant Value:
16
(0x00000010)
⑸ android LinearLayout 裡面的東西怎麼換行
由於前段時間項目中使用到了自動換行的線性布局,本來打算用表格布局在裡面一個個的用Java代碼添加ImageView的,但是添加的View控制項是不確定的,因為得靠伺服器的數據返回,就這樣手動用Java代碼畫布局的方式就這樣夭折了,因為在表哥布局中我無法確定一行顯示多少個ImageView的數目,所以無法動態添加,最後自能自己去看看那種能夠換行的線性布局了,線性布局比較不好的是不能自動換行,也就是當設置LinearLayout的orentation 設置為vertical 為豎直方向也就是只有一列,每行只能顯示一個View或者View的子類,當設置LinearLayout的orentitation為Horizontal,LinearLayout的只能顯示為一行,橫向顯示,當屏幕滿了的時候,View控制項並不會自動換行,所以我們要做的就是在LinearLayout滿的時候自動換行。
需要了解的是怎麼樣繪制根據子控制項的長寬繪制父控制項的寬度與高度,所以需要傳入的參數控制項的高度,視圖分為兩種一種是View類型的,代表控制項有TextView,Button,EditText 等等,還有一種是裝視圖的容器控制項繼承自ViewGroup的控制項,如LinearLayout,RelativeLayout,TabHost等等控制項,需要自動換行的線性布局的話,就需要根據子控制項的高度與寬度,來動態載入父控制項的高度與寬度,所以需要在構造函數中傳入每一個子控制項的固定的高度,或者是動態設置子控制項的高度與寬度。
將自定義的LinearLayout 也繼承自ViewGroup 並且重寫抽象類ViewGrouop的幾個方法:onMeasure(),onLayout(),dispathDraw() 三個方法的意思分別是:第一個onMeasure()是用來計算控制項以及子控制項所佔用的區域,第二個onLayout()是控制子控制項的換行,第三個可寫可不寫,主要是用來繪制控制項的邊框,
自定義LinearLayout的代碼如下:
[java] view plainprint?
package com.huanglong.mylinearlayout;
import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.graphics.Rect;
import android.util.AttributeSet;
import android.view.View;
import android.view.ViewGroup;
/**
* @author huanglong 2013-5-28 自定義自動換行LinearLayout
*/
public class FixGridLayout extends ViewGroup {
private int mCellWidth;
private int mCellHeight;
public FixGridLayout(Context context) {
super(context);
}
public FixGridLayout(Context context, AttributeSet attrs) {
super(context, attrs);
}
public FixGridLayout(Context context, AttributeSet attrs, int defStyle) {
super(context, attrs, defStyle);
}
public void setmCellWidth(int w) {
mCellWidth = w;
requestLayout();
}
public void setmCellHeight(int h) {
mCellHeight = h;
requestLayout();
}
/**
* 控制子控制項的換行
*/
@Override
protected void onLayout(boolean changed, int l, int t, int r, int b) {
int cellWidth = mCellWidth;
int cellHeight = mCellHeight;
int columns = (r - l) / cellWidth;
if (columns < 0) {
columns = 1;
}
int x = 0;
int y = 0;
int i = 0;
int count = getChildCount();
for (int j = 0; j < count; j++) {
final View childView = getChildAt(j);
// 獲取子控制項Child的寬高
int w = childView.getMeasuredWidth();
int h = childView.getMeasuredHeight();
// 計運算元控制項的頂點坐標
int left = x + ((cellWidth - w) / 2);
int top = y + ((cellHeight - h) / 2);
// int left = x;
// int top = y;
// 布局子控制項
childView.layout(left, top, left + w, top + h);
if (i >= (columns - 1)) {
i = 0;
x = 0;
y += cellHeight;
} else {
i++;
x += cellWidth;
}
}
}
/**
* 計算控制項及子控制項所佔區域
*/
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
// 創建測量參數
int cellWidthSpec = MeasureSpec.makeMeasureSpec(mCellWidth, MeasureSpec.AT_MOST);
int cellHeightSpec = MeasureSpec.makeMeasureSpec(mCellHeight, MeasureSpec.AT_MOST);
// 記錄ViewGroup中Child的總個數
int count = getChildCount();
// 設置子空間Child的寬高
for (int i = 0; i < count; i++) {
View childView = getChildAt(i);
/*
* 090 This is called to find out how big a view should be. 091 The
* parent supplies constraint information in the width and height
* parameters. 092 The actual mesurement work of a view is performed
* in onMeasure(int, int), 093 called by this method. 094 Therefore,
* only onMeasure(int, int) can and must be overriden by subclasses.
* 095
*/
childView.measure(cellWidthSpec, cellHeightSpec);
}
// 設置容器控制項所佔區域大小
// 注意setMeasuredDimension和resolveSize的用法
setMeasuredDimension(resolveSize(mCellWidth * count, widthMeasureSpec),
resolveSize(mCellHeight * count, heightMeasureSpec));
// setMeasuredDimension(widthMeasureSpec, heightMeasureSpec);
// 不需要調用父類的方法
// super.onMeasure(widthMeasureSpec, heightMeasureSpec);
}
/**
* 為控制項添加邊框
*/
@Override
protected void dispatchDraw(Canvas canvas) {
// 獲取布局控制項寬高
int width = getWidth();
int height = getHeight();
// 創建畫筆
Paint mPaint = new Paint();
// 設置畫筆的各個屬性
mPaint.setColor(Color.BLUE);
mPaint.setStyle(Paint.Style.STROKE);
mPaint.setStrokeWidth(10);
mPaint.setAntiAlias(true);
// 創建矩形框
Rect mRect = new Rect(0, 0, width, height);
// 繪制邊框
canvas.drawRect(mRect, mPaint);
// 最後必須調用父類的方法
super.dispatchDraw(canvas);
}
}
然後在Xml文件中引用自己定義的控制項,在Java代碼中調用:
[java] view plainprint?
package com.huanglong.mylinearlayout;
import android.os.Bundle;
import android.app.Activity;
import android.view.Menu;
import android.view.MenuItem;
import android.widget.CheckBox;
import android.widget.SimpleAdapter;
import android.support.v4.app.NavUtils;
public class MainActivity extends Activity {
private SimpleAdapter adapter;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
FixGridLayout fixGridLayout = (FixGridLayout) findViewById(R.id.ll);
fixGridLayout.setmCellHeight(30);
fixGridLayout.setmCellWidth(100);
for (int i = 0; i < 7; i++) {
CheckBox box = new CheckBox(MainActivity.this);
box.setText("第"+i+"個");
fixGridLayout.addView(box);
}
}
@Override
public boolean onCreateOptionsMenu(Menu menu) {
getMenuInflater().inflate(R.menu.activity_main, menu);
return true;
}
}