46

在我的 Galaxy Tab 10.1 上的日历应用程序中,当创建一个新事件时,会在标题栏/操作栏区域出现一个包含完成和取消按钮的对话框。

在此处输入图像描述

我想在我的应用程序中实现这一点。除了在我的子类setHasOptionsMenu(true)中覆盖之外,我还尝试过使用,但我的操作项没有出现。我也尝试过从内部调用,但它总是返回。onCreateOptionsMenuDialogFragmentgetDialog().getActionBar()onCreateViewnull

如果我启动一个Activity而不是显示一个对话框,但它占据了整个屏幕,我就可以让它工作。有没有标准的方法可以使用 a 来做到这一点DialogFragment

4

8 回答 8

86

使用来自谷歌小组帖子的想法,我能够将其从活动样式中拉出来。您最好将高度和宽度修改为您选择的“动态”大小。然后设置您想要的任何 ActionBar 按钮

<style name="PopupTheme" parent="android:Theme.Holo.Light.Dialog">
    <item name="android:windowIsFloating">false</item>
    <item name="android:windowContentOverlay">@null</item>
    <item name="android:windowSoftInputMode">stateAlwaysHidden</item>
    <item name="android:windowActionModeOverlay">true</item>
    <item name="android:windowIsTranslucent">true</item>
</style>

--

public static void showAsPopup(Activity activity) {
    //To show activity as dialog and dim the background, you need to declare android:theme="@style/PopupTheme" on for the chosen activity on the manifest
    activity.requestWindowFeature(Window.FEATURE_ACTION_BAR);
    activity.getWindow().setFlags(WindowManager.LayoutParams.FLAG_DIM_BEHIND,
            WindowManager.LayoutParams.FLAG_DIM_BEHIND);
    LayoutParams params = activity.getWindow().getAttributes(); 
    params.height = 850; //fixed height
    params.width = 850; //fixed width
    params.alpha = 1.0f;
    params.dimAmount = 0.5f;
    activity.getWindow().setAttributes((android.view.WindowManager.LayoutParams) params); 
    setContentView(R.layout.activity_main);
}
于 2012-09-05T05:31:40.997 回答
14

如果您使用的是 ActionBarSherlock,请按如下方式声明主题:

<style name="PopupTheme" parent="Theme.Sherlock">
    <item name="android:windowFrame">@null</item>
    <item name="android:windowIsFloating">false</item>
    <item name="android:windowContentOverlay">@null</item>
    <item name="android:windowAnimationStyle">@android:style/Animation.Dialog</item>
    <item name="android:windowSoftInputMode">stateAlwaysHidden</item>
    <item name="android:windowActionModeOverlay">true</item>
    <item name="android:colorBackgroundCacheHint">@null</item>
    <item name="android:windowCloseOnTouchOutside">true</item>
    <item name="android:windowIsTranslucent">true</item>
    <item name="windowContentOverlay">@null</item>
</style>

并且,根据Luke Sleeman 的回答使用 PopupTheme初始化SherlockActivity

private void showAsPopup(SherlockActivity activity) {
    //To show activity as dialog and dim the background, you need to declare android:theme="@style/PopupTheme" on for the chosen activity on the manifest
    //activity.requestWindowFeature(Window.FEATURE_ACTION_BAR); // NO NEED to call this line.
    activity.getWindow().setFlags(WindowManager.LayoutParams.FLAG_DIM_BEHIND,
            WindowManager.LayoutParams.FLAG_DIM_BEHIND);
    LayoutParams params = activity.getWindow().getAttributes(); 
    params.alpha = 1.0f;
    params.dimAmount = 0.5f;
    activity.getWindow().setAttributes((android.view.WindowManager.LayoutParams) params); 

    // This sets the window size, while working around the IllegalStateException thrown by ActionBarView
    activity.getWindow().setLayout(width,height);
}

结果:

在此处输入图像描述

于 2013-03-04T16:22:20.157 回答
11

在实施 StrikeForceZero 和 Luke Sleeman 建议的解决方案时遇到了一些麻烦,所以我想贡献我的经验。我敢肯定,我只是缺少一些东西,因此非常感谢您的反馈。

我所做的是以下内容:

  1. 使用提供的 PopupTheme 创建样式,直接复制/粘贴:

    <style name="PopupTheme" parent="android:Theme.Holo.Light.Dialog">
        <item name="android:windowIsFloating">false</item>
        <item name="android:windowContentOverlay">@null</item>
        <item name="android:windowSoftInputMode">stateAlwaysHidden</item>
        <item name="android:windowActionModeOverlay">true</item>
        <item name="android:windowIsTranslucent">true</item>
    </style>
    
  2. 添加 showAsPopup() 方法作为片段中的方法,该方法将打开假对话片段,直接复制/粘贴:

    private void showAsPopup(Activity activity) {
        //To show activity as dialog and dim the background, you need to declare android:theme="@style/PopupTheme" on for the chosen activity on the manifest
        activity.requestWindowFeature(Window.FEATURE_ACTION_BAR);
        activity.getWindow().setFlags(WindowManager.LayoutParams.FLAG_DIM_BEHIND,
            WindowManager.LayoutParams.FLAG_DIM_BEHIND);
        LayoutParams params = activity.getWindow().getAttributes(); 
        params.alpha = 1.0f;
        params.dimAmount = 0f;
        activity.getWindow().setAttributes((android.view.WindowManager.LayoutParams) params); 
    
        // This sets the window size, while working around the IllegalStateException thrown by ActionBarView
        activity.getWindow().setLayout(850,850);
    }
    
  3. 使用简单的 new() 调用创建新活动的实例,然后将其传递给 showAsPopup() 方法:

    DialogTestActivity test = new DialogTestActivity();
    showAsPopup(test);
    
  4. 为了测试的目的(我只是想确认我可以打开一个显示为带有操作栏的对话框的活动)我使用了一个非常简单的测试,直接从按钮视图 api 演示中窃取(用于布局文件,请参阅 api 演示中的buttons_1.xml):

    public class DialogTestActivity extends Activity {
    
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.buttons_test);
        }
    }
    

不幸的是,每次我尝试这个时,我都会在第一次调用时得到一个未指定的空指针异常,activity.requestWindowFeature(Window.FEATURE_ACTION_BAR);

04-29 16:39:05.361: W/System.err(15134): java.lang.NullPointerException
04-29 16:39:05.361: W/System.err(15134):    at android.app.Activity.requestWindowFeature(Activity.java:3244)
04-29 16:39:05.371: W/System.err(15134):    at packagenameremovedforlegalreasons.classname.showAsPopup(classname.java:602)
04-29 16:39:05.371: W/System.err(15134):    at packagenameremovedforlegalreasons.classname.onMapLongClick(classname.java:595)
04-29 16:39:05.371: W/System.err(15134):    at com.google.android.gms.maps.GoogleMap$5.onMapLongClick(Unknown Source)
04-29 16:39:05.371: W/System.err(15134):    at com.google.android.gms.internal.k$a.onTransact(Unknown Source)
04-29 16:39:05.381: W/System.err(15134):    at android.os.Binder.transact(Binder.java:310)
04-29 16:39:05.381: W/System.err(15134):    at com.google.android.gms.maps.internal.IOnMapLongClickListener$Stub$Proxy.onMapLongClick(IOnMapLongClickListener.java:93)
04-29 16:39:05.381: W/System.err(15134):    at maps.i.s.a(Unknown Source)
04-29 16:39:05.381: W/System.err(15134):    at maps.y.v.d(Unknown Source)
04-29 16:39:05.381: W/System.err(15134):    at maps.y.bf.onLongPress(Unknown Source)
04-29 16:39:05.381: W/System.err(15134):    at maps.d.v.onLongPress(Unknown Source)
04-29 16:39:05.381: W/System.err(15134):    at maps.d.h.c(Unknown Source)
04-29 16:39:05.381: W/System.err(15134):    at maps.d.h.c(Unknown Source)
04-29 16:39:05.381: W/System.err(15134):    at maps.d.j.handleMessage(Unknown Source)
04-29 16:39:05.391: W/System.err(15134):    at android.os.Handler.dispatchMessage(Handler.java:99)
04-29 16:39:05.391: W/System.err(15134):    at android.os.Looper.loop(Looper.java:137)
04-29 16:39:05.391: W/System.err(15134):    at android.app.ActivityThread.main(ActivityThread.java:5041)
04-29 16:39:05.391: W/System.err(15134):    at java.lang.reflect.Method.invokeNative(Native Method)
04-29 16:39:05.391: W/System.err(15134):    at java.lang.reflect.Method.invoke(Method.java:511)
04-29 16:39:05.391: W/System.err(15134):    at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:793)
04-29 16:39:05.391: W/System.err(15134):    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:560)
04-29 16:39:05.391: W/System.err(15134):    at dalvik.system.NativeStart.main(Native Method)

正如您从堆栈跟踪中看到的那样,预期的行为是在 GoogleMap 实例上长按打开窗口(使用 API 2 中的 MapFragments)。所以我的第一个想法是尝试从 Fragment 打开存在问题,所以我将调用传递回拥有的 Activity。同样的错误,同样没有附加信息。

在这一点上,我最好的猜测是 new() 调用没有充分实例化类/视图以进行调用以修改其视图。事实证明,这似乎至少在某种程度上是正确的,因为将视图修改代码迁移到活动中并以正常方式简单地打开活动是可行的:

调用活动:

    public void openMapDialog()
    {
        Intent intent = new Intent(this, DialogTestActivity.class);
        startActivity(intent);
    }

新班级代码:

    public class DialogTestActivity extends Activity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);

        // From: https://stackoverflow.com/questions/11425020/actionbar-in-a-dialogfragment
        this.requestWindowFeature(Window.FEATURE_ACTION_BAR);
        this.getWindow().setFlags(WindowManager.LayoutParams.FLAG_DIM_BEHIND, WindowManager.LayoutParams.FLAG_DIM_BEHIND);
        LayoutParams params = this.getWindow().getAttributes(); 
        params.alpha = 1.0f;
        params.dimAmount = 0f;
        this.getWindow().setAttributes((android.view.WindowManager.LayoutParams) params); 

        // This sets the window size, while working around the IllegalStateException thrown by ActionBarView
        this.getWindow().setLayout(600,600);

        setContentView(R.layout.buttons_test);
        }
    }

所以我想我发布所有这些的目的是为了澄清如果你想做上述海报建议的事情,你不能只是 new() 一个活动并调用 showAsPopup()。这可能是我对 Android 显示的缺乏经验,但是虽然这似乎有点明显,但将 showAsPopup() 解释为由当前视图调用而不是正在创建的视图调用似乎也很自然,因为您正在传递活动实例(如果它应该在 onCreate() 中完成,就像我最终做的那样)。

因此,如果意图是在创建活动而不是创建活动中调用 showAsPopup(),那么在调用 onCreate() 之前如何获取可修改的 Activity 实例并不明显。问题是您不能在调用 setContentView() 之后调用诸如 requestWindowFeature() 之类的东西(示例),这是一个问题,因为它通常在 onCreate() 中调用。

同样,如果有一种简单/更好的方法可以做到这一点,我将非常感谢反馈。希望这对想要使用这种方法的人有所帮助。

于 2013-04-29T21:58:39.927 回答
4

我花了很多时间来玩这个。接受的答案适用于galaxy nexus 7(Android 4.2),但在三星galaxy SIII(Android 4.1)和三星galaxy tab 10.2(Android 4.0)上失败,但有以下例外:

IllegalStateException: ActionBarView can only be used with android:layout_width="match_parent" (or fill_parent)

这是由ActionBarView.onMeasure(int, int)中的代码引起的,该代码检查以确保布局设置为 match_parent。正确的解决方案是通过 setLayout 而不是使用 setAttributes 来设置窗口的宽度。

这是 showAsPopup() 的固定版本,适用于我测试过的所有设备:

private void showAsPopup(Activity activity) {
    //To show activity as dialog and dim the background, you need to declare android:theme="@style/PopupTheme" on for the chosen activity on the manifest
    activity.requestWindowFeature(Window.FEATURE_ACTION_BAR);
    activity.getWindow().setFlags(WindowManager.LayoutParams.FLAG_DIM_BEHIND,
            WindowManager.LayoutParams.FLAG_DIM_BEHIND);
    LayoutParams params = activity.getWindow().getAttributes(); 
    params.alpha = 1.0f;
    params.dimAmount = 0f;
    activity.getWindow().setAttributes((android.view.WindowManager.LayoutParams) params); 

    // This sets the window size, while working around the IllegalStateException thrown by ActionBarView
    activity.getWindow().setLayout(850,850);
}

为了完整起见,这里又是 PopupTheme:

<style name="PopupTheme" parent="android:Theme.Holo.Light.Dialog">
    <item name="android:windowIsFloating">false</item>
    <item name="android:windowContentOverlay">@null</item>
    <item name="android:windowSoftInputMode">stateAlwaysHidden</item>
    <item name="android:windowActionModeOverlay">true</item>
    <item name="android:windowIsTranslucent">true</item>
</style>
于 2013-02-11T10:40:21.367 回答
1

就像 Veeti 暗示的那样,您可能想尝试使用 Dialog Theme 实现 Activity。在 Android 清单中:

<activity android:name=".YourActivity" android:theme="@android:style/Theme.Dialog </activity>

希望这会有所帮助。

于 2012-07-11T03:27:56.827 回答
1

我可能没有那么有经验,但我会使用一个活动,添加操作栏项目,然后:

<activity android:name=".YourActivity" android:theme="@android:style/Theme.Holo.Light.Dialog" />

希望有帮助!

于 2012-07-11T04:01:12.567 回答
1

我找到了一种很好的方法,在 DialogFragment 的 alertDialog 的构建器上使用setCustomTitle(也可以在 alertDialog 本身上使用)。

public Dialog onCreateDialog(final Bundle savedInstanceState) {
    final AlertDialog.Builder builder = new AlertDialog.Builder(activity,...);
    final Toolbar toolbar = (Toolbar) inflater.inflate(R.layout.dialog_app_filter_toolbar, null, false);
    // <= prepare toolbar and dialog here
    return builder.create();
    }

结果(来自我的应用程序):

在此处输入图像描述

在此处输入图像描述

如果您不想使用 AlertDialog,您仍然可以在对话框布局中放置一个工具栏并使用它。

于 2016-08-12T12:04:14.763 回答
0

我知道这不是真正的答案,但上面的解决方案似乎很不雅,以至于人们可能认为完全避免操作栏会更容易,比如从 .noActionBar 继承对话框主题,然后在面板顶部创建视图模仿动作栏,至少这样它都保留在 xlm 中。

于 2014-10-20T01:18:51.497 回答