oliviergの答えは私にとってうまくいき、カスタムDialogクラスを作成することが目的のルートである場合は、最良のソリューションです。しかし、AlertDialogクラスを使用できないことに悩みました。デフォルトのシステムAlertDialogスタイルを使用できるようにしたいと思いました。カスタムダイアログクラスの作成には、このスタイルはありません。
そこで、カスタムクラスを作成しなくても機能するソリューション(ハック)を見つけました。既存のビルダーを使用できます。
AlertDialogは、タイトルのプレースホルダーとしてコンテンツビューの上にビューを配置します。ビューを見つけて高さを0に設定すると、スペースはなくなります。
これまでに2.3と3.0でテストしましたが、まだすべてのバージョンで動作しない可能性があります。
これを行うための2つのヘルパーメソッドを次に示します。
/**
* Show a Dialog with the extra title/top padding collapsed.
*
* @param customView The custom view that you added to the dialog
* @param dialog The dialog to display without top spacing
* @param show Whether or not to call dialog.show() at the end.
*/
public static void showDialogWithNoTopSpace(final View customView, final Dialog dialog, boolean show) {
// Now we setup a listener to detect as soon as the dialog has shown.
customView.getViewTreeObserver().addOnGlobalLayoutListener(new OnGlobalLayoutListener() {
@Override
public void onGlobalLayout() {
// Check if your view has been laid out yet
if (customView.getHeight() > 0) {
// If it has been, we will search the view hierarchy for the view that is responsible for the extra space.
LinearLayout dialogLayout = findDialogLinearLayout(customView);
if (dialogLayout == null) {
// Could find it. Unexpected.
} else {
// Found it, now remove the height of the title area
View child = dialogLayout.getChildAt(0);
if (child != customView) {
// remove height
LinearLayout.LayoutParams lp = (LinearLayout.LayoutParams) child.getLayoutParams();
lp.height = 0;
child.setLayoutParams(lp);
} else {
// Could find it. Unexpected.
}
}
// Done with the listener
customView.getViewTreeObserver().removeGlobalOnLayoutListener(this);
}
}
});
// Show the dialog
if (show)
dialog.show();
}
/**
* Searches parents for a LinearLayout
*
* @param view to search the search from
* @return the first parent view that is a LinearLayout or null if none was found
*/
public static LinearLayout findDialogLinearLayout(View view) {
ViewParent parent = (ViewParent) view.getParent();
if (parent != null) {
if (parent instanceof LinearLayout) {
// Found it
return (LinearLayout) parent;
} else if (parent instanceof View) {
// Keep looking
return findDialogLinearLayout((View) parent);
}
}
// Couldn't find it
return null;
}
使用例は次のとおりです。
Dialog dialog = new AlertDialog.Builder(this)
.setView(yourCustomView)
.create();
showDialogWithNoTopSpace(yourCustomView, dialog, true);
これをDialogFragmentで使用している場合は、DialogFragmentのonCreateDialog
メソッドをオーバーライドします。次に、上記の最初の例のようにダイアログを作成して返します。唯一の変更点は、ダイアログでshow()を呼び出さないように、3番目のパラメーター(show)としてfalseを渡す必要があることです。DialogFragmentは後でそれを処理します。
例:
@Override
public Dialog onCreateDialog(Bundle savedInstanceState) {
Dialog dialog = new AlertDialog.Builder(getContext())
.setView(yourCustomView)
.create();
showDialogWithNoTopSpace(yourCustomView, dialog, false);
return dialog;
}
これをさらにテストするとき、必要な追加の微調整があれば必ず更新します。