我在正在构建的应用程序中遇到了这个问题。请忽略所有设计缺陷和缺乏最佳实践方法,这纯粹是为了展示我无法解决的问题。
我有 DialogFragment
,它返回一个基本的 AlertDialog
和使用 View
设置的自定义 AlertDialog.Builder.setView()
。如果此 View
有特定的大小要求,我如何让 Dialog
正确调整自身大小以显示自定义 View
中的所有内容?
这是我一直在使用的示例代码:
package com.test.test;
import android.os.Bundle;
import android.app.Activity;
import android.app.AlertDialog;
import android.app.Dialog;
import android.app.DialogFragment;
import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.graphics.Paint.Style;
import android.view.Gravity;
import android.view.LayoutInflater;
import android.view.View;
import android.view.WindowManager;
import android.view.View.OnClickListener;
import android.view.ViewGroup;
import android.view.ViewGroup.LayoutParams;
import android.widget.ArrayAdapter;
import android.widget.Button;
import android.widget.EditText;
import android.widget.FrameLayout;
import android.widget.LinearLayout;
import android.widget.Spinner;
import android.widget.TextView;
public class MainActivity extends Activity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
// Use a button for launching
Button b = new Button(this);
b.setText("Launch");
b.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
// Launch the dialog
myDialog d = new myDialog();
d.show(getFragmentManager(), null);
}
});
setContentView(b);
}
public static class myDialog extends DialogFragment {
@Override
public Dialog onCreateDialog(Bundle savedInstanceState) {
// Create the dialog
AlertDialog.Builder db = new AlertDialog.Builder(getActivity());
db.setTitle("Test Alert Dialog:");
db.setView(new myView(getActivity()));
return db.create();
}
protected class myView extends View {
Paint p = null;
public myView(Context ct) {
super(ct);
// Setup paint for the drawing
p = new Paint();
p.setColor(Color.MAGENTA);
p.setStyle(Style.STROKE);
p.setStrokeWidth(10);
}
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
setMeasuredDimension(800, 300);
}
@Override
protected void onDraw(Canvas canvas) {
// Draw a rectangle showing the bounds of the view
canvas.drawRect(0, 0, getMeasuredWidth(), getMeasuredHeight(), p);
}
}
}
}
创建了一个 Button
,单击即可打开 DialogFragment
。自定义 View
(myView
) 需要具有 800 的宽度和 300 的高度,这在 onMeasure()
的覆盖中正确设置。此 View
以洋红色绘制其测量范围以进行调试。
800 的宽度比我设备上的默认 Dialog
尺寸宽,但被剪裁而不是正确拉伸(stretch)。
我查看了以下解决方案:
- DialogFragment.getDialog returns null
- How to control the width and height of the default Alert Dialog in Android?
- Size of Alert Dialog or Custom Alert Dialog
我推导出了以下两种编码方式:
- 获取
WindowManager.LayoutParams
的Dialog
并使用myDialog.getDialog().getWindow().get/setAttributes()
覆盖它们 - 通过
setLayout(w, h)
使用myDialog.getDialog().getWindow().setLayout()
方法
我在所有我能想到的地方都尝试过它们(覆盖 onStart()
,在 onShowListener
中,在创建和显示 Dialog
之后等)并且如果为 LayoutParams
提供 特定值,通常可以使这两种方法正常工作。但是只要提供 WRAP_CONTENT
,什么都不会发生。
有什么建议吗?
编辑:
情况截图:
特定值的屏幕截图(注意在此处输入 900,850 并未覆盖 View 的整个宽度,鉴于正在调整整个窗口,这是有道理的。因此,如果需要另一个,则提供了 WRAP_CONTENT
的原因是必要的/固定值不合适):
最佳答案
我有一个可行的解决方案,老实说,我认为挖掘得太深而无法获得如此简单的结果。但这里是:
到底发生了什么:
通过使用 Hierarchy Viewer 打开 Dialog
布局,我能够检查 AlertDialog
的整个布局以及到底发生了什么:
蓝色高亮部分是所有高级部分(Window
、Dialog
视觉样式的框架等),从末尾开始blue 向下是 AlertDialog
的组件所在的位置(red = 标题,yellow = ScrollView stub , 可能用于列表 AlertDialog
s, green = Dialog
内容,即自定义 View ,orange = 按钮)。
从这里很明显,7 View 路径(从 蓝色 的开头到 绿色 的结尾)是未能正确WRAP_CONTENT
。查看每个 View
的 LayoutParams.width
显示所有都被赋予了 LayoutParams.width = MATCH_PARENT
并且某处(我猜在顶部)a大小已设置。所以如果你跟随那棵树,很明显你自定义的 View
在树的底部,永远不会能够影响 Dialog< 的大小
。
那么现有的解决方案在做什么?
- 我的问题中提到的两种编码方法都是简单地获取顶部
View
并修改其LayoutParams
。显然,在树中的所有View
对象与父对象匹配的情况下,如果顶层设置为静态大小,则整个Dialog
将改变大小。但如果顶层设置为WRAP_CONTENT
,则树中所有其余的View
对象仍然查找树到“MATCH他们的 parent ”,而不是向下看以“包装他们的内容”。
如何解决问题:
说白了就是把影响路径中所有View
对象的LayoutParams.width
改为WRAP_CONTENT
。
我发现这只能在调用 DialogFragment
的 onStart
生命周期步骤之后完成。所以 onStart
的实现如下:
@Override
public void onStart() {
// This MUST be called first! Otherwise the view tweaking will not be present in the displayed Dialog (most likely overriden)
super.onStart();
forceWrapContent(myCustomView);
}
然后函数来适当修改View
层次结构LayoutParams
:
protected void forceWrapContent(View v) {
// Start with the provided view
View current = v;
// Travel up the tree until fail, modifying the LayoutParams
do {
// Get the parent
ViewParent parent = current.getParent();
// Check if the parent exists
if (parent != null) {
// Get the view
try {
current = (View) parent;
} catch (ClassCastException e) {
// This will happen when at the top view, it cannot be cast to a View
break;
}
// Modify the layout
current.getLayoutParams().width = LayoutParams.WRAP_CONTENT;
}
} while (current.getParent() != null);
// Request a layout to be re-done
current.requestLayout();
}
这是工作结果:
这让我感到困惑,为什么整个 Dialog
不希望成为 WRAP_CONTENT
并设置显式 minWidth
来处理适合默认大小,但我确信它是有充分理由的(有兴趣听到它)。
关于java - 具有自定义 View : Resize to wrap the view's content 的 AlertDialog,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/14907104/