android動態設置高度
1. Android 自定義控制項 動態設置高度
Android動態改變View控制項大小的方法:
1、聲明控制項參數獲取對象 LayoutParams lp;
2、獲取控制項參數: lp = 控制項id.getLayoutParams();
3、設置控制項參數:如高度。 lp.height -= 10;
4:、使設置生效:控制項id.setLayoutParams(lp);
例如如要把Imageview下移200px: ImageView.setPadding( ImageView.getPaddingLeft(), ImageView.getPaddingTop()+200, ImageView.getPaddingRight(), ImageView.getPaddingBottom());
2. 有哪些android開發技巧
1、android:clipToPadding
意思是控制項的繪制區域是否在padding裡面。默認為true。如果你設置了此屬性值為false,就能實現一個在布局上事半功陪的效果。先看一個效果圖。
上圖中的ListView頂部默認有一個間距,向上滑動後,間距消失,如下圖所示。
如果使用margin或padding,都不能實現這個效果。加一個headerView又顯得大材小用,而且過於麻煩。此處的clipToPadding配合paddingTop效果就剛剛好。
同樣,還有另外一個屬性也很神奇:android:clipChildren,具體請參考:【Android】神奇的android:clipChildren屬性
2、match_parent和wrap_content
按理說這兩個屬性一目瞭然,一個是填充布局空間適應父控制項,一個是適應自身內容大小。但如果在列表如ListView中,用錯了問題就大了。ListView中的getView方法需要計算列表條目,那就必然需要確定ListView的高度,onMesure才能做測量。如果指定了wrap_content,就等於告訴系統,如果我有一萬個條目,你都幫我計算顯示出來,然後系統按照你的要求就new了一萬個對象出來。那你不悲劇了?先看一個圖。
假設現在ListView有8條數據,match_parent需要new出7個對象,而wrap_content則需要8個。這里涉及到View的重用,就不多探討了。所以這兩個屬性的設置將決定getView的調用次數。
由此再延伸出另外一個問題:getView被多次調用。
什麼叫多次調用?比如position=0它可能調用了幾次。看似很詭異吧。GridView和ListView都有可能出現,說不定這個禍首就是wrap_content。說到底是View的布局出現了問題。如果嵌套的View過於復雜,解決方案可以是通過代碼測量列表所需要的高度,或者在getView中使用一個小技巧:parent.getChildCount == position
@Override
public View getView(int position, View convertView, ViewGroup parent) {
if (parent.getChildCount() == position) {
// does things here
}
return convertView;
}
3、IllegalArgumentException: pointerIndex out of range
出現這個Bug的場景還是很無語的。一開始我用ViewPager + PhotoView(一個開源控制項)顯示圖片,在多點觸控放大縮小時就出現了這個問題。一開始我懷疑是PhotoView的bug,找了半天無果。要命的是不知如何try,老是crash。後來才知道是android遺留下來的bug,源碼里沒對pointer index做檢查。改源碼重新編譯不太可能吧。明知有exception,又不能從根本上解決,如果不讓它crash,那就只能try-catch了。解決辦法是:自定義一個ViewPager並繼承ViewPager。請看以下代碼:
/**
* 自定義封裝android.support.v4.view.ViewPager,重寫onInterceptTouchEvent事件,捕獲系統級別異常
*/
public class CustomViewPager extends ViewPager {
public CustomViewPager(Context context) {
this(context, null);
}
public CustomViewPager(Context context, AttributeSet attrs) {
super(context, attrs);
}
@Override
public boolean onInterceptTouchEvent(MotionEvent ev) {
try {
return super.onInterceptTouchEvent(ev);
} catch (IllegalArgumentException e) {
LogUtil.e(e);
} catch ( e) {
LogUtil.e(e);
}
return false;
}
}
把用到ViewPager的布局文件,替換成CustomViewPager就OK了。
4、ListView中item點擊事件無響應
listView的Item點擊事件突然無響應,問題一般是在listView中加入了button、checkbox等控制項後出現的。這個問題是聚焦沖突造成的。在android裡面,點擊屏幕之後,點擊事件會根據你的布局來進行分配的,當你的listView裡面增加了button之後,點擊事件第一優先分配給你listView裡面的button。所以你的點擊Item就失效了,這個時候你就要根據你的需求,是給你的item的最外層layout設置點擊事件,還是給你的某個布局元素添加點擊事件了。
解決辦法:在ListView的根控制項中設置(若根控制項是LinearLayout, 則在LinearLayout中加入以下屬性設置)descendantFocusability屬性。
android:descendantFocusability="blocksDescendants"
官方文檔也是這樣說明。
5、getSupportFragmentManager()和getChildFragmentManager()
有一個需求,Fragment需要嵌套3個Fragment。基本上可以想到用ViewPager實現。開始代碼是這樣寫的:
mViewPager.setAdapter(new CustomizeFragmentPagerAdapter(getActivity().getSupportFragmentManager(), subFragmentList));
導致的問題是嵌套的Fragment有時會莫名其妙不顯示。開始根本不知道問題出現在哪,當你不知道問題的原因時,去解決這個問題顯然比較麻煩。經過一次又一次的尋尋覓覓,終於在stackoverflow上看到了同樣的提問。說是用getChildFragmentManager()就可以了。真是這么神奇!
mViewPager.setAdapter(new CustomizeFragmentPagerAdapter(getChildFragmentManager, subFragmentList));
讓我們看一下這兩個有什麼區別。首先是getSupportFragmentManager(或者getFragmentManager)的說明:
Return the FragmentManager for interacting with fragments associated with this fragment's activity.
然後是getChildFragmentManager:
Return a private FragmentManager for placing and managing Fragments inside of this Fragment.
Basically, the difference is that Fragment's now have their own internal FragmentManager that can handle Fragments. The child FragmentManager is the one that handles Fragments contained within only the Fragment that it was added to. The other FragmentManager is contained within the entire Activity.
已經說得比較明白了。
6、ScrollView嵌套ListView
這樣的設計是不是很奇怪?兩個同樣會滾動的View居然放到了一起,而且還是嵌套的關系。曾經有一個這樣的需求:界面一共有4個區域部分,分別是公司基本信息(logo、名稱、法人、地址)、公司簡介、公司榮譽、公司口碑列表。每部分內容都需要根據內容自適應高度,不能寫死。鄙人首先想到的也是外部用一個ScrollView包圍起來。然後把這4部分分別用4個自定義控制項封裝起來。基本信息和公司簡介比較簡單,榮譽需要用到RecyclerView和TextView的組合,RecyclerView(當然,用GridView也可以,3列多行的顯示)存放榮譽圖片,TextView顯示榮譽名稱。最後一部分口碑列表當然是ListView了。這時候,問題就出來了。需要解決ListView放到ScrollView中的滑動問題和RecyclerView的顯示問題(如果RecyclerView的高度沒法計算,你是看不到內容的)。
當然,網上已經有類似的提問和解決方案了。
給一個網址:
四種方案解決ScrollView嵌套ListView問題
ListView的情況還比較好解決,優雅的做法無非寫一個類繼承ListView,然後重寫onMeasure方法。
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
int expandSpec = MeasureSpec.makeMeasureSpec(Integer.MAX_VALUE >> 2, MeasureSpec.AT_MOST);
super.onMeasure(widthMeasureSpec, expandSpec);
}
ListView可以重寫onMeasure解決,RecyclerView重寫這個方法是行不通的。
說到底其實計算高度嘛。有兩種方式,一種是動態計算RecycleView,然後設置setLayoutParams;另外一種跟ListView的解決方式類似,定義一個類繼承LinearLayoutManager或GridLayoutManager(注意:可不是繼承RecyclerView),重寫onMeasure方法(此方法比較麻煩,此處不表,下次寫一篇文章再作介紹)。
動態計算高度如下:
int heightPx = DensityUtil.dip2px(getActivity(), (imageHeight + imageRowHeight) * lines);
MarginLayoutParams mParams = new MarginLayoutParams(LayoutParams.MATCH_PARENT, heightPx);
mParams.setMargins(0, 0, 0, 0);
LinearLayout.LayoutParams lParams = new LinearLayout.LayoutParams(mParams);
honorImageRecyclerView.setLayoutParams(lParams);
思路是這樣的:服務端返回榮譽圖片後,由於是3列顯示的方式,只需要計算需要顯示幾行,然後給定行間距和圖片的高度,再設置setLayoutParams就行了。
int lines = (int) Math.ceil(totalImages / 3d);
至此,這個奇怪的需求得到了解決。
可是在滑動的時候,感覺出現卡頓的現象。聰明的你肯定想到是滑動沖突了。應該是ScrollView的滑動干擾到了ListView的滑動。怎麼辦呢?能不能禁掉ScrollView的滑動?
網路一下,你肯定能搜索到答案的。先上代碼:
/**
* @author Leo
*
* Created in 2015-9-12
* 攔截ScrollView滑動事件
*/
public class CustomScrollView extends ScrollView {
private int downY;
private int touchSlop;
public CustomScrollView(Context context) {
this(context, null);
}
public CustomScrollView(Context context, AttributeSet attrs) {
this(context, attrs, 0);
}
public CustomScrollView(Context context, AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
touchSlop = ViewConfiguration.get(context).getScaledTouchSlop();
}
@Override
public boolean onInterceptTouchEvent(MotionEvent e) {
int action = e.getAction();
switch (action) {
case MotionEvent.ACTION_DOWN:
downY = (int) e.getRawY();
break;
case MotionEvent.ACTION_MOVE:
int moveY = (int) e.getRawY();
if (Math.abs(moveY - downY) > touchSlop) {
return true;
}
}
return super.onInterceptTouchEvent(e);
}
}
只要理解了getScaledTouchSlop()這個方法就好辦了。這個方法的注釋是:Distance in pixels a touch can wander before we think the user is scrolling。說這是一個距離,表示滑動的時候,手的移動要大於這個距離才開始移動控制項,如果小於此距離就不觸發移動。
看似很完美了。
但是還有另外一個問題:我每次載入這個界面花的時間太長了,每次由其它界面啟動這個界面時,都要卡上1~2秒,而且因手機性能時間不等。並不是由於網路請求,取數據由子線程做,跟UI線程毫無關系。這樣的體驗自己看了都很不爽。
幾天過去了,還是那樣。馬上要給老闆演示了。這樣的體驗要被罵十次呀。
難道跟ScrollView的嵌套有關?
好吧,那我重構代碼。不用ScrollView了。直接用一個ListView,然後add一個headerView存放其它內容。因為控制項封裝得還算好,沒改多少布局就OK了,一運行,流暢順滑,一切迎刃而解!
本來就是這么簡單的問題,為什麼非得用ScrollView嵌套呢?
stackoverflow早就告訴你了,不要這樣嵌套!不要這樣嵌套!不要這樣嵌套!重要的事情說三遍。
ListView inside ScrollView is not scrolling on Android
當然,從android 5.0 Lollipop開始提供了一種新的API支持嵌入滑動,此時,讓像這樣的需求也能很好實現。
此處給一個網址,大家有興趣自行了解,此處不再討論。
Android NestedScrolling 實戰
7、EmojiconTextView的setText(null)
這是開源表情庫com.rockerhieu.emojicon中的TextView加強版。相信很多人用到過這個開源工具包。TextView用setText(null)完全沒問題。但EmojiconTextView setText(null)後就悲劇了,直接crash,顯示的是null pointer。開始我懷疑時這個view沒初始化,但並不是。那就調試一下唄。
@Override
public void setText(CharSequence text, BufferType type) {
SpannableStringBuilder builder = new SpannableStringBuilder(text);
EmojiconHandler.addEmojis(getContext(), builder, mEmojiconSize);
super.setText(builder, type);
}
EmojiconTextView中的setText看來沒什麼問題。點SpannableStringBuilder進去看看,源碼原來是這樣的:
/**
* Create a new SpannableStringBuilder containing a of the
* specified text, including its spans if any.
*/
public SpannableStringBuilder(CharSequence text) {
this(text, 0, text.length());
}
好吧。問題已經找到了,text.length(),不空指針才怪。
text = text == null ? "" : text;
SpannableStringBuilder builder = new SpannableStringBuilder(text);
加一行判斷就行了。
先想到這么多,以後再補充。
3. android 如何動態設置控制項的寬度和高度
android中的控制項如果在xml布局文件中把控制項的layout_width和layout_height寫成固定值了,好像就不能再在程序中更改該控制項的高度和寬度了,不知哪位大俠有何良策可以指教一二,如 xml文件內容如下: <LinearLayout android:id="@id/dialog_bottom_neutral" android:layout_width="fill_parent" android:layout_height="wrap_content" android:layout_weight="1.0" android:gravity="center"<Buttonandroid:id="@id/dialog_bottom_neutral_button" android:layout_width="80.0dip" android:layout_height="28.0dip" android:background="@drawable/dlg_button" android:gravity="center" android:singleLine="true" android:text="mid" android:textColor="@drawable/dlg_button_text_color" android:textSize="14.0sp" / 小弟在程序中使用button.width和button.height設置沒用,用LayoutParmas設置也沒用
4. android recyclerview 怎麼給item 設置高度
一種是在item的布局文件里設置固定的高度;還有就是代碼動態設置,通過LayoutParame去設置。
5. android LinearLayout 裡面的東西怎麼換行
由於前段時間項目中使用到了自動換行的線性布局,本來打算用表格布局在裡面一個個的用java代碼添加ImageView的,但是添加的View控制項是不確定的,因為得靠伺服器的數據返回,就這樣手動用Java代碼畫布局的方式就這樣夭折了,因為在表哥布局中我無法確定一行顯示多少個ImageView的數目,所以無法動態添加,最後自能自己去看看那種能夠換行的線性布局了,線性布局比較不好的是不能自動換行,也就是當設置LinearLayout的orentation 設置為vertical 為豎直方向也就是只有一列,每行只能顯示一個View或者View的子類,當設置LinearLayout的orentitation為Horizontal,LinearLayout的只能顯示為一行,橫向顯示,當屏幕滿了的時候,View控制項並不會自動換行,所以我們要做的就是在LinearLayout滿的時候自動換行。x0dx0a需要了解的是怎麼樣繪制根據子控制項的長寬繪制父控制項的寬度與高度,所以需要傳入的參數控制項的高度,視圖分為兩種一種是View類型的,代表控制項有TextView,Button,EditText 等等,還有一種是裝視圖的容器控制項繼承自ViewGroup的控制項,如LinearLayout,RelativeLayout,TabHost等等控制項,需要自動換行的線性布局的話,就需要根據子控制項的高度與寬度,來動態載入父控制項的高度與寬度,所以需要在構造函數中傳入每一個子控制項的固定的高度,或者是動態設置子控制項的高度與寬度。x0dx0a將自定義的LinearLayout 也繼承自ViewGroup 並且重寫抽象類ViewGrouop的幾個方法:onMeasure(),onLayout(),dispathDraw() 三個方法的意思分別是:第一個onMeasure()是用來計算控制項以及子控制項所佔用的區域,第二個onLayout()是控制子控制項的換行,第三個可寫可不寫,主要是用來繪制控制項的邊框,x0dx0a自定義LinearLayout的代碼如下:x0dx0ax0dx0a[java] view plainprint?x0dx0apackage com.huanglong.mylinearlayout; x0dx0a x0dx0aimport android.content.Context; x0dx0aimport android.graphics.Canvas; x0dx0aimport android.graphics.Color; x0dx0aimport android.graphics.Paint; x0dx0aimport android.graphics.Rect; x0dx0aimport android.util.AttributeSet; x0dx0aimport android.view.View; x0dx0aimport android.view.ViewGroup; x0dx0a x0dx0a/** x0dx0a * @author huanglong 2013-5-28 自定義自動換行LinearLayout x0dx0a */ x0dx0apublic class FixGridLayout extends ViewGroup { x0dx0a private int mCellWidth; x0dx0a private int mCellHeight; x0dx0a x0dx0a public FixGridLayout(Context context) { x0dx0a super(context); x0dx0a } x0dx0a x0dx0a public FixGridLayout(Context context, AttributeSet attrs) { x0dx0a super(context, attrs); x0dx0a } x0dx0a x0dx0a public FixGridLayout(Context context, AttributeSet attrs, int defStyle) { x0dx0a super(context, attrs, defStyle); x0dx0a } x0dx0a x0dx0a public void setmCellWidth(int w) { x0dx0a mCellWidth = w; x0dx0a requestLayout(); x0dx0a } x0dx0a x0dx0a public void setmCellHeight(int h) { x0dx0a mCellHeight = h; x0dx0a requestLayout(); x0dx0a } x0dx0a x0dx0a /** x0dx0a * 控制子控制項的換行 x0dx0a */ x0dx0a @Override x0dx0a protected void onLayout(boolean changed, int l, int t, int r, int b) { x0dx0a int cellWidth = mCellWidth; x0dx0a int cellHeight = mCellHeight; x0dx0a int columns = (r - l) / cellWidth; x0dx0a if (columns < 0) { x0dx0a columns = 1; x0dx0a } x0dx0a int x = 0; x0dx0a int y = 0; x0dx0a int i = 0; x0dx0a int count = getChildCount(); x0dx0a for (int j = 0; j < count; j++) { x0dx0a final View childView = getChildAt(j); x0dx0a // 獲取子控制項Child的寬高 x0dx0a int w = childView.getMeasuredWidth(); x0dx0a int h = childView.getMeasuredHeight(); x0dx0a // 計運算元控制項的頂點坐標 x0dx0a int left = x + ((cellWidth - w) / 2); x0dx0a int top = y + ((cellHeight - h) / 2); x0dx0a // int left = x; x0dx0a // int top = y; x0dx0a // 布局子控制項 x0dx0a childView.layout(left, top, left + w, top + h); x0dx0a x0dx0a if (i >= (columns - 1)) { x0dx0a i = 0; x0dx0a x = 0; x0dx0a y += cellHeight; x0dx0a } else { x0dx0a i++; x0dx0a x += cellWidth; x0dx0a x0dx0a } x0dx0a } x0dx0a } x0dx0a x0dx0a /** x0dx0a * 計算控制項及子控制項所佔區域 x0dx0a */ x0dx0a @Override x0dx0a protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { x0dx0a // 創建測量參數 x0dx0a int cellWidthSpec = MeasureSpec.makeMeasureSpec(mCellWidth, MeasureSpec.AT_MOST); x0dx0a int cellHeightSpec = MeasureSpec.makeMeasureSpec(mCellHeight, MeasureSpec.AT_MOST); x0dx0a // 記錄ViewGroup中Child的總個數 x0dx0a int count = getChildCount(); x0dx0a // 設置子空間Child的寬高 x0dx0a for (int i = 0; i < count; i++) { x0dx0a View childView = getChildAt(i); x0dx0a /* x0dx0a * 090 This is called to find out how big a view should be. 091 The x0dx0a * parent supplies constraint information in the width and height x0dx0a * parameters. 092 The actual mesurement work of a view is performed x0dx0a * in onMeasure(int, int), 093 called by this method. 094 Therefore, x0dx0a * only onMeasure(int, int) can and must be overriden by subclasses. x0dx0a * 095 x0dx0a */ x0dx0a childView.measure(cellWidthSpec, cellHeightSpec); x0dx0a } x0dx0a // 設置容器控制項所佔區域大小 x0dx0a // 注意setMeasuredDimension和resolveSize的用法 x0dx0a setMeasuredDimension(resolveSize(mCellWidth * count, widthMeasureSpec), x0dx0a resolveSize(mCellHeight * count, heightMeasureSpec)); x0dx0a // setMeasuredDimension(widthMeasureSpec, heightMeasureSpec); x0dx0a x0dx0a // 不需要調用父類的方法 x0dx0a // super.onMeasure(widthMeasureSpec, heightMeasureSpec); x0dx0a } x0dx0a x0dx0a /** x0dx0a * 為控制項添加邊框 x0dx0a */ x0dx0a @Override x0dx0a protected void dispatchDraw(Canvas canvas) { x0dx0a // 獲取布局控制項寬高 x0dx0a int width = getWidth(); x0dx0a int height = getHeight(); x0dx0a // 創建畫筆 x0dx0a Paint mPaint = new Paint(); x0dx0a // 設置畫筆的各個屬性 x0dx0a mPaint.setColor(Color.BLUE); x0dx0a mPaint.setStyle(Paint.Style.STROKE); x0dx0a mPaint.setStrokeWidth(10); x0dx0a mPaint.setAntiAlias(true); x0dx0a // 創建矩形框 x0dx0a Rect mRect = new Rect(0, 0, width, height); x0dx0a // 繪制邊框 x0dx0a canvas.drawRect(mRect, mPaint); x0dx0a // 最後必須調用父類的方法 x0dx0a super.dispatchDraw(canvas); x0dx0a } x0dx0a x0dx0a} x0dx0a然後在Xml文件中引用自己定義的控制項,在Java代碼中調用:x0dx0ax0dx0a[java] view plainprint?x0dx0apackage com.huanglong.mylinearlayout; x0dx0a x0dx0aimport android.os.Bundle; x0dx0aimport android.app.Activity; x0dx0aimport android.view.Menu; x0dx0aimport android.view.MenuItem; x0dx0aimport android.widget.CheckBox; x0dx0aimport android.widget.SimpleAdapter; x0dx0aimport android.support.v4.app.NavUtils; x0dx0a x0dx0apublic class MainActivity extends Activity { x0dx0a private SimpleAdapter adapter; x0dx0a @Override x0dx0a public void onCreate(Bundle savedInstanceState) { x0dx0a super.onCreate(savedInstanceState); x0dx0a setContentView(R.layout.activity_main); x0dx0a FixGridLayout fixGridLayout = (FixGridLayout) findViewById(R.id.ll); x0dx0a fixGridLayout.setmCellHeight(30); x0dx0a fixGridLayout.setmCellWidth(100); x0dx0a for (int i = 0; i < 7; i++) { x0dx0a CheckBox box = new CheckBox(MainActivity.this); x0dx0a box.setText("第"+i+"個"); x0dx0a fixGridLayout.addView(box); x0dx0a } x0dx0a } x0dx0a x0dx0a @Override x0dx0a public boolean onCreateOptionsMenu(Menu menu) { x0dx0a getMenuInflater().inflate(R.menu.activity_main, menu); x0dx0a return true; x0dx0a } x0dx0ax0dx0a}