16

所以我在使用 ActionBarSherlock 并决定切换到新的 ActionBarCompat。使用 ABS,可以使用本文中描述的方式隐藏 ActionBar: 如何在创建活动之前隐藏操作栏,然后再次显示?

但是,使用 ActionBarCompat,应用程序在 API14 上崩溃,因为当您设置android:windowActionBar为方法时返回 null,即使您已false在方法中声明了。getSupportActionBar()getWindow().requestFeature(Window.FEATURE_ACTION_BAR);onCreate()

有趣的是,如果您getActionBar()改为调用,您将获得对象并且一切正常。

那么,这是一个错误还是我错过了什么?欢迎任何想法!


styles.xml文件:

<style name="Theme.MyApp" parent="@style/Theme.AppCompat.Light.DarkActionBar">
    <item name="android:windowActionBar">false</item>
    <item name="android:windowTitleSize">0dp</item>
</style>

MyActivity.java文件:

...
@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    // Get the action bar feature. This feature is disabled by default into the theme
    // for specific reasons.
    getWindow().requestFeature(Window.FEATURE_ACTION_BAR);
    ...
    // By default the action bar is hidden.
    getSupportActionBar().hide();
}
4

4 回答 4

19

我遇到了同样的问题,在我看来,找到了这种奇怪行为的原因。我查看了支持库的来源并得到了这个:

Appcompat在ActionBarActivityDelegatemHasActionBar中创建新的操作栏之前检查变量的值

final ActionBar getSupportActionBar() {
    // The Action Bar should be lazily created as mHasActionBar or mOverlayActionBar
    // could change after onCreate
    if (mHasActionBar || mOverlayActionBar) {
        if (mActionBar == null) {
            mActionBar = createSupportActionBar();
    ...

我们可以通过调用supportRequestWindowFeature(int featureId)委托ActionBarActivityActionBarActivityDelegate来更改它的值。

有基础委托类ActionBarDelegateBase及其后代ActionBarDelegateHC,,,,根据运行的android版本选择其中一个ActionBarActivityDelegateICSActionBarActivityJB并且方法supportRequestWindowFeature实际上几乎在所有这些中都可以正常工作,但是它被ActionBarActivityDelegateICS这样覆盖了

@Override
public boolean supportRequestWindowFeature(int featureId) {
    return mActivity.requestWindowFeature(featureId);
}

所以它对变量没有影响mHasActionBar,这就是为什么getSupportActionBar()返回null。

我们快到了。我来到了两种不同的解决方案。

第一种方式

  1. 从git导入 appcompat 的源项目

  2. 将覆盖的方法更改ActionBarActivityDelegateICS.java为这样的东西

    @Override
    public boolean supportRequestWindowFeature(int featureId) {
        boolean result = mActivity.requestWindowFeature(featureId);
        if (result) {
            switch (featureId) {
            case WindowCompat.FEATURE_ACTION_BAR:
                mHasActionBar = true;
            case WindowCompat.FEATURE_ACTION_BAR_OVERLAY:
                mOverlayActionBar = true;
            }
        }
        return result;
    }
    
  3. 将此行放在活动的onCreate方法之前getSupportActionBar()

    supportRequestWindowFeature(WindowCompat.FEATURE_ACTION_BAR);
    

第二种方式

  1. 从 android SDK 导入 appcompat 项目(src 目录为空)

  2. 将此方法添加到您的活动中

    private void requestFeature() {
        try {
            Field fieldImpl = ActionBarActivity.class.getDeclaredField("mImpl");
            fieldImpl.setAccessible(true);
            Object impl = fieldImpl.get(this);
    
            Class<?> cls = Class.forName("android.support.v7.app.ActionBarActivityDelegate");
    
            Field fieldHasActionBar = cls.getDeclaredField("mHasActionBar");
            fieldHasActionBar.setAccessible(true);
            fieldHasActionBar.setBoolean(impl, true);
    
        } catch (NoSuchFieldException e) {
            Log.e(LOG_TAG, e.getLocalizedMessage(), e);
        } catch (IllegalAccessException e) {
            Log.e(LOG_TAG, e.getLocalizedMessage(), e);
        } catch (IllegalArgumentException e) {
            Log.e(LOG_TAG, e.getLocalizedMessage(), e);
        } catch (ClassNotFoundException e) {
            Log.e(LOG_TAG, e.getLocalizedMessage(), e);
        }
    }
    
  3. 像这样调用您requestFeature()onCreate活动方法

    if (Build.VERSION.SDK_INT >= 11) {
        requestFeature();
    }
    supportRequestWindowFeature(WindowCompat.FEATURE_ACTION_BAR);
    

我使用了第二种方式。就这样。

于 2013-12-01T18:09:30.583 回答
5

我用它来隐藏 AppCompat 中的 ActionBar:style.xml

<style name="Theme.AppCompat.Light.NoActionBar" parent="@style/Theme.AppCompat.Light">
    <item name="android:windowNoTitle">true</item>
</style>

AndroidManifest.xml:

<activity
        android:name=".Splash"
        android:label="@string/title_activity_splash"
        android:theme="@style/Theme.AppCompat.Light.NoActionBar">
        <intent-filter>
            <action android:name="android.intent.action.MAIN" />
            <category android:name="android.intent.category.LAUNCHER" />
        </intent-filter>
</activity>
于 2014-06-26T20:00:23.950 回答
0

我不知道我是否完全理解你的问题,但我走了。

我认为您需要同时使用:getSupportActionBar()旧版本和getActionBar()最新版本。这不是错误。

在使用这些方法之前,您需要验证设备版本。

我希望我能提供帮助。

于 2013-09-20T13:13:53.997 回答
0

您的活动是否扩展了 ActionBarActivity?可能它没有,这就是它使用 getActionbar() 方法运行的原因。

于 2013-09-20T14:37:59.623 回答