gpt4 book ai didi

java - 具有自定义 View : Resize to wrap the view's content 的 AlertDialog

转载 作者:IT老高 更新时间:2023-10-28 20:26:38 25 4
gpt4 key购买 nike

我在正在构建的应用程序中遇到了这个问题。请忽略所有设计缺陷和缺乏最佳实践方法,这纯粹是为了展示我无法解决的问题。

我有 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)。

我查看了以下解决方案:

我推导出了以下两种编码方式:

  1. 获取 WindowManager.LayoutParamsDialog 并使用 myDialog.getDialog().getWindow().get/setAttributes() 覆盖它们
  2. 通过setLayout(w, h)使用myDialog.getDialog().getWindow().setLayout()方法

我在所有我能想到的地方都尝试过它们(覆盖 onStart() ,在 onShowListener 中,在创建和显示 Dialog 之后等)并且如果为 LayoutParams 提供 特定值,通常可以使这两种方法正常工作。但是只要提供 WRAP_CONTENT,什么都不会发生。

有什么建议吗?

编辑:

情况截图: enter image description here

特定值的屏幕截图(注意在此处输入 900,850 并未覆盖 View 的整个宽度,鉴于正在调整整个窗口,这是有道理的。因此,如果需要另一个,则提供了 WRAP_CONTENT 的原因是必要的/固定值不合适): enter image description here

最佳答案

我有一个可行的解决方案,老实说,我认为挖掘得太深而无法获得如此简单的结果。但这里是:

到底发生了什么:

通过使用 Hierarchy Viewer 打开 Dialog 布局,我能够检查 AlertDialog 的整个布局以及到底发生了什么: enter image description here

蓝色高亮部分是所有高级部分(WindowDialog 视觉样式的框架等),从末尾开始blue 向下是 AlertDialog 的组件所在的位置(red = 标题,yellow = ScrollView stub , 可能用于列表 AlertDialogs, green = Dialog 内容,即自定义 View ,orange = 按钮)。

从这里很明显,7 View 路径(从 蓝色 的开头到 绿色 的结尾)是未能正确WRAP_CONTENT。查看每个 ViewLayoutParams.width 显示所有都被赋予了 LayoutParams.width = MATCH_PARENT 并且某处(我猜在顶部)a大小已设置。所以如果你跟随那棵树,很明显你自定义的 View 在树的底部,永远不会能够影响 Dialog< 的大小

那么现有的解决方案在做什么?

  • 我的问题中提到的两种编码方法都是简单地获取顶部View并修改其LayoutParams。显然,在树中的所有 View 对象与父对象匹配的情况下,如果顶层设置为静态大小,则整个 Dialog 将改变大小。但如果顶层设置为 WRAP_CONTENT,则树中所有其余的 View 对象仍然查找树到“MATCH他们的 parent ”,而不是向下看以“包装他们的内容”。

如何解决问题:

说白了就是把影响路径中所有View对象的LayoutParams.width改为WRAP_CONTENT

我发现这只能在调用 DialogFragmentonStart 生命周期步骤之后完成。所以 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();
}

这是工作结果: enter image description here

这让我感到困惑,为什么整个 Dialog 不希望成为 WRAP_CONTENT 并设置显式 minWidth 来处理适合默认大小,但我确信它是有充分理由的(有兴趣听到它)。

关于java - 具有自定义 View : Resize to wrap the view's content 的 AlertDialog,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/14907104/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com