ConstraintLayout 中的 Gone 元素会留下幽灵空间
Posted
技术标签:
【中文标题】ConstraintLayout 中的 Gone 元素会留下幽灵空间【英文标题】:Gone elements in ConstraintLayout leaves ghost space 【发布时间】:2019-02-27 03:43:34 【问题描述】:所以我有一个 RecycleView
的布局文件,它具有可展开/可折叠的视图。
点击标题会展开/折叠额外的数据。在编辑器中一切看起来都很好。但是,它会有一个类似于layout_marginBottom
的幽灵空间。
首次加载的外观
扩展
正确折叠
所以编辑器只会显示ViewHolder
,如图 2 和图 3 所示。
但是在设备上运行时,它首先会显示图 1,然后单击它展开后会显示图 2。再次折叠它会显示图 3 而不是图 1。它将继续显示正确的数字(2 和 3)。
图 2 中的额外内容具有共同的父项,其标头为 ConstraintLayout
。
无论GONE
说明符如何,都会发生这种情况:以编程方式或 XML
使用activity.runOnUIThread
无济于事。使用new Handler().onPostDelayed
只会像图 1 那样更糟,但会出现更多不需要的空间,与图 2 一样多,只是没有额外的信息。
这是 XML 文件:
<?xml version="1.0" encoding="utf-8"?>
<android.support.v7.widget.CardView xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
android:layout_
android:layout_
android:layout_marginBottom="10dp"
app:cardCornerRadius="10dp">
<android.support.constraint.ConstraintLayout
android:layout_
android:layout_ >
<LinearLayout
android:id="@+id/header"
android:layout_
android:layout_
android:gravity="center_vertical"
android:orientation="horizontal"
android:padding="10dp"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintEnd_toStartOf="@+id/validity"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toTopOf="parent"
app:layout_constraintVertical_bias="0.0">
<ImageButton
android:id="@+id/expand"
android:layout_
android:layout_
android:background="@android:color/transparent"
android:layout_marginRight="8dp"
android:layout_marginEnd="8dp"
app:srcCompat="@drawable/ic_right" />
<TextView
android:id="@+id/title"
android:layout_
android:layout_
android:layout_weight="1"
android:ellipsize="end"
android:maxLines="1"
android:textColor="@color/textDark" />
</LinearLayout>
<View
android:id="@+id/divider"
android:layout_
android:layout_
android:layout_marginEnd="16dp"
android:layout_marginLeft="16dp"
android:layout_marginRight="16dp"
android:layout_marginStart="16dp"
android:background="@color/colorSecondaryDark"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toBottomOf="@+id/header" />
<TextView
android:id="@+id/date"
android:layout_
android:layout_
android:layout_marginEnd="8dp"
android:layout_marginLeft="10dp"
android:layout_marginRight="8dp"
android:layout_marginStart="10dp"
android:layout_marginTop="8dp"
app:layout_constraintEnd_toStartOf="@+id/size"
app:layout_constraintHorizontal_bias="0.0"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toBottomOf="@+id/divider" />
<TextView
android:id="@+id/size"
android:layout_
android:layout_
android:layout_marginEnd="16dp"
android:layout_marginRight="16dp"
android:layout_marginTop="8dp"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintTop_toBottomOf="@+id/divider" />
<TextView
android:id="@+id/amount"
android:layout_
android:layout_
android:layout_marginEnd="8dp"
android:layout_marginLeft="10dp"
android:layout_marginRight="8dp"
android:layout_marginStart="10dp"
android:layout_marginTop="8dp"
app:layout_constraintEnd_toStartOf="@+id/validity"
app:layout_constraintHorizontal_bias="0.0"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toBottomOf="@+id/date" />
<LinearLayout
android:id="@+id/validity"
android:layout_
android:layout_
android:layout_marginEnd="16dp"
android:layout_marginRight="16dp"
android:layout_marginTop="8dp"
android:gravity="center_vertical"
android:orientation="horizontal"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintTop_toBottomOf="@+id/size">
<ImageView
android:id="@+id/help"
android:layout_
android:layout_
android:layout_marginEnd="8dp"
android:layout_marginRight="8dp"
android:layout_weight="1"
android:contentDescription="@string/backup_layout_help_desc"
app:srcCompat="@drawable/ic_help" />
<TextView
android:id="@+id/invalid"
android:layout_
android:layout_
android:layout_weight="1"
android:text="@string/backup_layout_invalid"
android:textAllCaps="true"
android:textColor="@color/red" />
</LinearLayout>
<TextView
android:id="@+id/passwords"
android:layout_
android:layout_
android:layout_marginEnd="10dp"
android:layout_marginStart="10dp"
android:layout_marginTop="8dp"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintHorizontal_bias="0.0"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toBottomOf="@+id/amount" />
<Button
android:id="@+id/delete"
android:layout_
android:layout_
android:layout_marginEnd="8dp"
android:layout_marginLeft="10dp"
android:layout_marginRight="8dp"
android:layout_marginStart="10dp"
android:layout_marginTop="10dp"
android:background="@android:color/transparent"
android:text="@string/backup_layout_delete"
android:textColor="@color/red"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintEnd_toStartOf="@+id/importshare"
app:layout_constraintHorizontal_bias="0.0"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toBottomOf="@+id/passwords"
app:layout_constraintVertical_bias="1.0" />
<LinearLayout
android:id="@+id/importshare"
android:layout_
android:layout_
android:layout_marginEnd="10dp"
android:layout_marginRight="10dp"
android:layout_marginTop="10dp"
android:orientation="horizontal"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintTop_toBottomOf="@+id/validity"
app:layout_constraintVertical_bias="1.0">
<Button
android:id="@+id/commit"
android:layout_
android:layout_
android:layout_weight="1"
android:background="@android:color/transparent"
android:text="Import"
android:textColor="@color/yellow" />
<ImageButton
android:id="@+id/share"
android:layout_
android:layout_
android:layout_weight="1"
android:background="@android:color/transparent"
android:contentDescription="Share"
android:padding="10dp"
app:srcCompat="@drawable/ic_share" />
</LinearLayout>
</android.support.constraint.ConstraintLayout>
以及 ViewHolder(剪切以显示相关部分):
public ViewHolder(View itemView, Context context)
super(itemView);
Amount = itemView.findViewById(R.id.amount);
Date = itemView.findViewById(R.id.date);
Delete = itemView.findViewById(R.id.delete);
Divider = itemView.findViewById(R.id.divider);
ExpandCollapse = itemView.findViewById(R.id.expand);
Import = itemView.findViewById(R.id.commit);
ImportShare = itemView.findViewById(R.id.importshare);
Header = itemView.findViewById(R.id.header);
Help = itemView.findViewById(R.id.help);
Passwords = itemView.findViewById(R.id.passwords);
Share = itemView.findViewById(R.id.share);
Size = itemView.findViewById(R.id.size);
Title = itemView.findViewById(R.id.title);
Validity = itemView.findViewById(R.id.validity);
Title.setTypeface(Typeface.createFromAsset(context.getAssets(), "cera.otf"));
Header.setOnClickListener(this);
onClick(Header);
@Override
public void onClick(View v)
if (expanded) collapse();
else expand();
public void expand()
ExpandCollapse.animate().rotation(90).start();
Date.setVisibility(View.VISIBLE);
Size.setVisibility(View.VISIBLE);
Amount.setVisibility(View.VISIBLE);
Passwords.setVisibility(View.VISIBLE);
Divider.setVisibility(View.VISIBLE);
Validity.setVisibility(valid ? View.INVISIBLE : View.VISIBLE);
Delete.setVisibility(View.VISIBLE);
ImportShare.setVisibility(View.VISIBLE);
expanded = true;
public void collapse()
ExpandCollapse.animate().rotation(0).start();
Date.setVisibility(View.GONE);
Size.setVisibility(View.GONE);
Amount.setVisibility(View.GONE);
Passwords.setVisibility(View.GONE);
Divider.setVisibility(View.GONE);
Validity.setVisibility(View.GONE);
Delete.setVisibility(View.GONE);
ImportShare.setVisibility(View.GONE);
expanded = false;
【问题讨论】:
如果 ConstraintLayout 的目的是消除嵌套,为什么还要嵌套 LinearLayouts? 标题需要 LinearLayouts 但其他的,现在你说它是不必要的。我只是不知道如何在 ConstraintLayouts 中水平居中。 使用约束和链。 我无法通过阅读代码来判断,但我敢打赌,如果您为所有视图临时设置不同的背景颜色(不是白色),您会看到哪个视图占用了该空间。 原来有两个视图显示为不可见,尽管它被指定为GONE
。不知道为什么,但是在修改适配器以从不将它们设置为INVISIBLE
和GONE
之后,一切都变得很好。顺便说一句,这两个视图是 TextView
和 ImageView
。
【参考方案1】:
科特林
我在键盘打开时尝试隐藏元素时遇到了非常相似的问题
当您打开键盘时会发生两次调整大小,一次调整用于打开和键盘,另一次用于将视图设置为消失。问题是第二次调整只会在另一个元素请求布局时才会发生,因此您的屏幕会非常明显地跳动两次
-
键盘打开时
当您点击或移动另一个 UI 元素时
我已经找到了解决方案,但只有当窗口软输入模式 = 调整调整大小时
见下文
class MainActivity : AppCompatActivity()
private var activityHeight = 0
private var keyboardOpen = false
private var mainScreen : ConstraintLayout? = null
private var infoText : TextView? = null
override fun onCreate(savedInstanceState: Bundle?)
super.onCreate(savedInstanceState)
setContentView(R.layout.main_screen)
mainScreen = findViewById(R.id.mainConstraintLayout)
infoText = findViewById(R.id.mainInfoText)
//...
/* Capture initial screen size */
this@MainActivity.window.decorView.doOnNextLayout
val displayFrame : Rect = Rect()
this@MainActivity.window.decorView.getWindowVisibleDisplayFrame(displayFrame)
activityHeight = displayFrame.height()
/* Check for keyboard open/close */
this@MainActivity.window.decorView.addOnLayoutChangeListener v, left, top, right, bottom, oldLeft, oldTop, oldRight, oldBottom ->
val drawFrame : Rect = Rect()
this@MainActivity.window.decorView.getWindowVisibleDisplayFrame(drawFrame)
val currentSize = drawFrame.height()
if(keyboardOpen)
keyboardOpen = currentSize < activityHeight
if (!keyboardOpen)
infoText?.visibility = View.VISIBLE
else
keyboardOpen = currentSize < activityHeight
if(keyboardOpen)
infoText?.visibility = View.GONE
/* Request a new layout on your base layout so screen adjusts correctly */
mainScreen?.requestLayout()
【讨论】:
以上是关于ConstraintLayout 中的 Gone 元素会留下幽灵空间的主要内容,如果未能解决你的问题,请参考以下文章
Recyclerview 中的 view.GONE 仍然保留空间
ConstraintLayout 中的 ListView 不显示