一、讓子View顯示超出自身容器大小
在子View的父布局上設(shè)置android:clipChildren=“false”,在View會(huì)通過這個(gè)方法判斷是否需要clip。使用的時(shí)候需要注意,這個(gè)參數(shù)只能設(shè)置在ViewGroup上,而生效的范圍是ViewGroup內(nèi)的子View。
void setDisplayListProperties(RenderNode renderNode) {
if (renderNode != null) {
...
renderNode.setClipToBounds(mParent instanceof ViewGroup
&& ((ViewGroup) mParent).getClipChildren());
...
}
....
}
沒有添加android:clipChildren=“false”,圖1的imagview是無(wú)法把圖片顯示完整
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent">
<ImageView
android:id="@+id/image"
android:layout_width="0dp"
android:layout_height="100dp"
android:src="@drawable/snow"
android:scaleType="centerCrop"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toTopOf="parent" />
</androidx.constraintlayout.widget.ConstraintLayout>

添加android:clipChildren=“false”后,圖2的imagview的大小是沒有改變,但是可以把超出imagview范圍的圖片內(nèi)容都顯示出來。

二、取消RecyclerView滑動(dòng)到邊緣上的水紋提示
RecyclerView是繼承ViewGroup的,而VIewGroup在初始的時(shí)候會(huì)設(shè)置WILL_NOT_DRAW進(jìn)行繪制的優(yōu)化,因?yàn)橐话鉜iewGroup只是作為容器用,具體的內(nèi)容應(yīng)該由View來繪制。
private void initViewGroup() {
// ViewGroup doesn't draw by default
if (!debugDraw()) {
setFlags(WILL_NOT_DRAW, DRAW_MASK);
}
...
}
當(dāng)然也有特殊情況,例如RecyclerView的滑動(dòng)到邊緣的水紋提示,這個(gè)就需要RecyclerView自身繪制,所以View也提供setWillNotDraw(boolean willNotDraw)這個(gè)方法對(duì)這個(gè)參數(shù)進(jìn)行修改。
public void setWillNotDraw(boolean willNotDraw) {
setFlags(willNotDraw ? WILL_NOT_DRAW : 0, DRAW_MASK);
}
所以RecyclerView也會(huì)調(diào)用這個(gè)方法進(jìn)行設(shè)置,果然在構(gòu)造里調(diào)用了setWillNotDraw(boolean willNotDraw)這個(gè)方法。
public RecyclerView(@NonNull Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
...
setWillNotDraw(getOverScrollMode() == View.OVER_SCROLL_NEVER);
...
}
需要取消水紋效果,就需要用setWillNotDraw(boolean willNotDraw)這個(gè)方法設(shè)置為true,而當(dāng)getOverScrollMode() == View.OVER_SCROLL_NEVER為true的時(shí)候就不會(huì)在繪制水紋了。而這個(gè)getOverScrollMode()方法是View里自帶的一個(gè)方法,所以應(yīng)該還有設(shè)置的方法。
在View里面可以找到這個(gè)設(shè)置方法,可以通過在代碼里設(shè)置了這個(gè)方法就可以把水紋取消。
public void setOverScrollMode(int overScrollMode) {
if (overScrollMode != OVER_SCROLL_ALWAYS &&
overScrollMode != OVER_SCROLL_IF_CONTENT_SCROLLS &&
overScrollMode != OVER_SCROLL_NEVER) {
throw new IllegalArgumentException("Invalid overscroll mode " + overScrollMode);
}
mOverScrollMode = overScrollMode;
}
其實(shí)除了可以在代碼上設(shè)置之外,還可以通過xml布局的時(shí)候設(shè)置,android:overScrollMode="never"
...
case R.styleable.View_overScrollMode:
overScrollMode = a.getInt(attr, OVER_SCROLL_IF_CONTENT_SCROLLS);
break;
...
三、狀態(tài)欄的設(shè)置
在style主題中設(shè)置這個(gè) <item name="android:windowTranslucentStatus">true</item> 會(huì)導(dǎo)致內(nèi)容顯示到狀態(tài)欄中。
隱藏狀態(tài)欄的方法 window.decorView.windowInsetsController?.hide(WindowInsets.Type.systemBars())
在劉海屏隱藏狀態(tài)欄,會(huì)導(dǎo)致狀態(tài)欄顯示黑色,可以用下面代碼解決
/**
* 適配劉海屏
* Fits notch screen.
*/
private fun fitsNotchScreen() {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.P) {
val lp: WindowManager.LayoutParams = window.getAttributes()
lp.layoutInDisplayCutoutMode = WindowManager.LayoutParams.LAYOUT_IN_DISPLAY_CUTOUT_MODE_SHORT_EDGES
window.setAttributes(lp)
}
}