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

标签 java android android-layout android-alertdialog android-dialogfragment

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

我有 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/

相关文章:

java - 如何忽略 PDFBox 2.0.7 使用的字体中缺失的字形

java - 流损坏异常 - 无效代码类型 AC [java]

java - 在 IntelliJ Idea 中,我的应用程序可以运行,但作为 jar 运行——不是因为 "Error: Could not find or load main class"

Android AppBarLayout 折叠工具栏并立即显示稀松布颜色

java - 如何将 android ImageView 捕捉到屏幕上的某个位置?

java - 理解 HashMap 中的 `structural modification`

java - MediaRecorder 不保存文件

java - 无法在未在警报对话框线程上调用 Looper.prepare() 的线程内创建处理程序

java - Android Java - 下拉微调器始终设置相同的文本,只想更改背景颜色

javascript - React Native - 轮式选择器 : Invariant violation. 'WheelCurvedPicker' 的 native 组件不存在