您好,登錄后才能下訂單哦!
本篇文章為大家展示了怎么在Android中動態設置字體大小,內容簡明扼要并且容易理解,絕對能使你眼前一亮,通過這篇文章的詳細介紹希望你能有所收獲。
在java代碼中取在dimens.xml中定義的值一共有三種:
getDimension()
getDimensionPixelOffset()
getDimensionPixelSize()
getDimension()
/** * Retrieve a dimensional for a particular resource ID. Unit * conversions are based on the current {@link DisplayMetrics} associated * with the resources. * * @param id The desired resource identifier, as generated by the aapt * tool. This integer encodes the package, type, and resource * entry. The value 0 is an invalid identifier. * * @return Resource dimension value multiplied by the appropriate * metric. */ public float getDimension(@DimenRes int id) throws NotFoundException { }
通過注釋我們不難發現,getDimension()是根據指定id獲取一個基于當前DisplayMetrics的值。這個值究竟是什么也沒有說,只知道是float,并且單位轉換是基于當前資源的,但肯定不是像素,如果是像素應該是int。
getDimensionPixelSize
/** * Retrieve a dimensional for a particular resource ID for use * as a size in raw pixels. This is the same as * {@link #getDimension}, except the returned value is converted to * integer pixels for use as a size. A size conversion involves * rounding the base value, and ensuring that a non-zero base value * is at least one pixel in size. * * @param id The desired resource identifier, as generated by the aapt * tool. This integer encodes the package, type, and resource * entry. The value 0 is an invalid identifier. * * @return Resource dimension value multiplied by the appropriate * metric and truncated to integer pixels. */ public int getDimensionPixelSize(@DimenRes int id) throws NotFoundException { }
getDimensionPixelSize()的功能與getDimension()類似,不同的是將結果轉換為int,并且小數部分四舍五入,這個結果將作為尺寸。getDimensionPixelSize()進行了尺寸轉換,這個轉換實際是上四舍五入的結果,并且保證返回值是一個至少是1像素的非零數值。
getDimensionPixelOffset()
/** * Retrieve a dimensional for a particular resource ID for use * as an offset in raw pixels. This is the same as * {@link #getDimension}, except the returned value is converted to * integer pixels for you. An offset conversion involves simply * truncating the base value to an integer. * * @param id The desired resource identifier, as generated by the aapt * tool. This integer encodes the package, type, and resource * entry. The value 0 is an invalid identifier. * * @return Resource dimension value multiplied by the appropriate * metric and truncated to integer pixels. */ public int getDimensionPixelOffset(@DimenRes int id) throws NotFoundException { }
getDimensionPixelOffset()與getDimension()功能類似,不同的是將結果轉換為int,這個結果將用作原始像素的偏移量。偏移轉換(offset conversion,函數命名中的offset是這個意思)的作用之一是將基礎值簡單地截短為整數,注意直接截斷小數位,即取整(其實就是把float強制轉化為int,注意不是四舍五入)。
階段性總結
由此可見,這三個函數返回的都是絕對尺寸,而不是相對尺寸(dp\sp等)。如果getDimension()返回結果是30.5f,那么getDimensionPixelSize()返回結果就是31,getDimensionPixelOffset()返回結果就是30。
至此,應該說getDimensionPixelSize() getDimension() getDimensionPixelOffset()我們已經大致有所了解了,但是如果想更深入了解一下,就需要深入源碼以驗證上述解釋。
扒源碼
深入源碼,我們可以發現其實這三個函數的實現大同小異,以getDimension()
為例:
public float getDimension(@DimenRes int id) throws NotFoundException { final TypedValue value = obtainTempTypedValue(); try { final ResourcesImpl impl = mResourcesImpl; impl.getValue(id, value, true); if (value.type == TypedValue.TYPE_DIMENSION) { return TypedValue.complexToDimension(value.data, impl.getDisplayMetrics()); } throw new NotFoundException("Resource ID #0x" + Integer.toHexString(id) + " type #0x" + Integer.toHexString(value.type) + " is not valid"); } finally { releaseTempTypedValue(value); } }
類TypedValue是動態類型數據的容器,其主要用于盛放Resources的值。上述代碼第7行就是根據id獲取TypedValue的值,getDimension()、getDimensionPixelOffset()和getDimensionPixelSize()函數體唯一的不同就是第7行:
getDimension()調用的是TypedValue的complexToDimension()方法
getDimensionPixelSize調用的是TypedValue的complexToDimensionPixelSize()方法
getDimensionPixelOffset調用的是TypedValue的complexToDimensionPixelOffset()方法
順藤摸瓜,我們繼續深入ypedValue,查看complexToDimension()、complexToDimensionPixelSize()和complexToDimensionPixelOffset()函數的區別,會發現這三個函數體內容依舊大同小異,以complexToDimension()為例:
public static float complexToDimension(int data, DisplayMetrics metrics) { return applyDimension( (data>>COMPLEX_UNIT_SHIFT)&COMPLEX_UNIT_MASK, complexToFloat(data), metrics); }
complexToDimensionPixelOffset()與complexToDimension()不同的是將結果進行了強轉,實際上相當直接截斷小數部分;
complexToDimensionPixelSize()是將結果進行四舍五入,并取整。這里的四舍五入實際上就是把結果加上0.5f然后進行強轉。
applyDimension()
各位看官,源碼已經看到了這里,是否已感覺很無趣?但applyDimension()的實現已經脫光了在等著你呢:
public static float applyDimension(int unit, float value,DisplayMetrics metrics) { switch (unit) { case COMPLEX_UNIT_PX: return value; case COMPLEX_UNIT_DIP: return value * metrics.density; case COMPLEX_UNIT_SP: return value * metrics.scaledDensity; case COMPLEX_UNIT_PT: return value * metrics.xdpi * (1.0f/72); case COMPLEX_UNIT_IN: return value * metrics.xdpi; case COMPLEX_UNIT_MM: return value * metrics.xdpi * (1.0f/25.4f); } return 0; }
在上述代碼中,我們發現在applyDimension()中根據單位的不同,將float乘上不同的系數。如dip/dp需乘上屏幕系數,sp則需乘上字號的縮放系數,pt、in、mm等也是根據相應的算法進行換算(從COMPLEX_UNIT_PX直接返回float可以看出,該方法是將數值轉成像素數)。
再次總結
通過上述探索,我們不難發現,在Adroid并沒有在java代碼中直接獲取dimens.xml中定義的dp(dip)/sp的值的API,只有getDimension()、getDimensionPixelOffset()和getDimensionPixelSize()這個三個方法來獲取絕對尺寸。但有時候我們確實需要動態獲取dimen.xml中的值,并為TextView設置字體大小。而這種方法直接應用在textView.setTextSize(dimen);都是有問題的。那我們將從TextView入手,尋找一個正確的姿勢來設置字體大小。
setTextSize()
首先把代碼端上來:
public void setTextSize(float size) { setTextSize(TypedValue.COMPLEX_UNIT_SP, size); }
原來setTextSize(float)調用了他的重載方法setTextSize(int,float),并且第一個參數傳的默認值是TypedValue.COMPLEX_UNIT_SP,眼熟嗎,沒錯就是之前提到的。那么,我們繼續看看一下setTextSize(int,float)做了什么:
public void setTextSize(int unit, float size) { if (!isAutoSizeEnabled()) { setTextSizeInternal(unit, size, true /* shouldRequestLayout */); } }
很顯然是調用了setTextSizeInternal(unit, size, true /* shouldRequestLayout */);。看到這累不,不過看都看了就再看看唄,說不定比蒼老師好看:
private void setTextSizeInternal(int unit, float size, boolean shouldRequestLayout) { Context c = getContext(); Resources r; if (c == null) { r = Resources.getSystem(); } else { r = c.getResources(); } setRawTextSize(TypedValue.applyDimension(unit, size, r.getDisplayMetrics()),shouldRequestLayout); }
高能!!!TypedValue.applyDimension(unit, size, r.getDisplayMetrics())是不是很眼熟???還記得applyDimension()是怎么處理數據的嗎?
我們發現在applyDimension()中根據單位的不同,將float乘上不同的系數。如dip/dp需乘上屏幕系數,sp則需乘上字號的縮放系數,pt、in、mm等也是根據相應的算法進行換算(從COMPLEX_UNIT_PX直接返回float可以看出,該方法是將數值轉成像素數)
綜上,setTextSize(float)給傳的值的單位其實是SP,但通過getDimension()取的值卻不是這樣的。為了證實默認單位是SP,各位看官可以直接傳個16,看看和16sp是不是一樣的。所以問題是不得到了解決?
結論
Android中并不提供直接從dimens.xml獲取dp/sp數值的方法,通過getDimensionPixelSize() getDimension() getDimensionPixelOffset()獲取的值是經過處理的。所以正確地動態設置TextView字體大小的姿勢應該是:
int dimen = getResources().getDimensionPixelSize(R.dimen.text_size); textView.setTextSize(TypedValue.COMPLEX_UNIT_PX,dimen);
上述內容就是怎么在Android中動態設置字體大小,你們學到知識或技能了嗎?如果還想學到更多技能或者豐富自己的知識儲備,歡迎關注億速云行業資訊頻道。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。