Android控件設(shè)置寬高比的方法
0. 困擾很久的問(wèn)題
Android控件的寬和高保持比例,這是從我接觸Android以來(lái),一直不斷會(huì)遇到的需求。以前,要么就是在代碼里直接設(shè)置寬和高,要么就是自定義控件。網(wǎng)上也有開(kāi)源的自定義ViewGroup,可以讓其子View比較方便的設(shè)置寬和高的比例。但這些實(shí)現(xiàn)方式,還是比較麻煩,也不夠直觀。直到有了DataBinding,我們可以很方便地給控件加上自定義的屬性,也就可以很方便的在布局文件中設(shè)置控件的寬高比了。
1. 如何實(shí)現(xiàn)
通過(guò)BinderAdapter為所有View綁定下面的方法,當(dāng)設(shè)置widthHeightRatio屬性時(shí),會(huì)調(diào)用下面這個(gè)方法。這個(gè)有點(diǎn)AOP的意思,我們針對(duì)所有的View做了處理。
public class DataBindingAdapters { // 根據(jù)View的高度和寬高比,設(shè)置高度 @BindingAdapter("widthHeightRatio") public static void setWidthHeightRatio(final View view, final float ratio) { view.getViewTreeObserver().addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() { @Override public void onGlobalLayout() { int height = view.getHeight(); if (height > 0) { view.getLayoutParams().width = (int) (height * ratio); view.invalidate(); view.getViewTreeObserver().removeGlobalOnLayoutListener(this); } } }); } }
我們?cè)讷@取到控件的高度后,根據(jù)比例計(jì)算寬度,然后設(shè)置給控件。這里注冊(cè)了OnGlobalLayoutListener,是因?yàn)榭丶母叨扔锌赡苓€沒(méi)計(jì)算完成。在獲取到高度之后,移除監(jiān)聽(tīng),避免多余的調(diào)用。
<ImageView android:layout_width="120dp" android:layout_height="match_parent" app:widthHeightRatio="@{1}"/>
然后我們就可以在布局文件中直接設(shè)置寬高比了。這個(gè)布局文件必須使用DataBinding,也就是最外層要用layout標(biāo)簽。屬性值必須加上@{},不然是按普通屬性處理的,不會(huì)調(diào)用我們的方法,編譯時(shí)會(huì)因?yàn)檎也坏綄傩詧?bào)錯(cuò)。當(dāng)然,這個(gè)屬性只能根據(jù)高度計(jì)算寬度,如果要根據(jù)寬度計(jì)算高度,可以用同樣的方式再加一個(gè)屬性。
2. 原理簡(jiǎn)析
其實(shí)在編譯后的layout文件中是沒(méi)有我們加的屬性的(編譯后的layout文件在build/intermediates/data-binding-layout-out下面可以看到)。真正設(shè)置這個(gè)屬性,還是在Java代碼中直接調(diào)用了我們綁定的方法。在DataBinding自動(dòng)生成的Binding類(lèi)中,可以發(fā)現(xiàn)有類(lèi)似下面這樣的調(diào)用。
DataBindingAdapters.setWidthHeightRatio(this.mboundView0, cardWidthHeightRatio);
這里只是做一個(gè)簡(jiǎn)單的解釋?zhuān)劣谠谑裁磿r(shí)機(jī)會(huì)觸發(fā)這行代碼,敬請(qǐng)期待我后續(xù)的文章。
3. BinderAdapter的其他妙用
ImageView自動(dòng)加載網(wǎng)絡(luò)圖片
@BindingAdapter({"android:src", "error"}) public static void setImageUrl(ImageView view, String url, @DrawableRes int errorImage) { if (!TextUtils.isEmpty(url)) { // 封裝好的圖片加載工具 ImageManager.from(view.getContext()).displayImage(view, url, errorImage); } else { view.setImageResource(errorImage); } }
直接在布局文件中設(shè)置要加載的圖片的url,和加載失敗時(shí)顯示的默認(rèn)圖。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Android開(kāi)發(fā)之計(jì)算器GridLayout布局實(shí)現(xiàn)方法示例
這篇文章主要介紹了Android開(kāi)發(fā)之計(jì)算器GridLayout布局實(shí)現(xiàn)方法,結(jié)合實(shí)例形式分析了Android計(jì)算器界面布局及表達(dá)式計(jì)算相關(guān)操作技巧,需要的朋友可以參考下2019-03-03Android基礎(chǔ)之Fragment與Activity交互詳解
以下小編就為大家介紹一下Fragment跟Activity之間的關(guān)系。需要的朋友可以過(guò)來(lái)參考下2013-07-07Android?NotificationListenerService?通知服務(wù)原理解析
這篇文章主要為大家介紹了Android?NotificationListenerService?通知服務(wù)原理解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-11-11Android抓取CSDN首頁(yè)極客頭條內(nèi)容完整實(shí)例
這篇文章主要介紹了Android抓取CSDN首頁(yè)極客頭條內(nèi)容完整實(shí)例,具有一定借鑒價(jià)值,需要的朋友可以參考下2018-01-01Android 8.0系統(tǒng)中應(yīng)用圖標(biāo)的適配微技巧
這篇文章主要介紹了Android 8.0系統(tǒng)中應(yīng)用圖標(biāo)的適配微技巧 ,需要的朋友可以參考下2018-04-04淺談Android Studio 4.1 更新內(nèi)容
這篇文章主要介紹了淺談Android Studio 4.1 更新內(nèi)容,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-10-10