0

背景:

我有一个使用 DrawerLayout 的 FragmentActivity。我创建了一个名为 NavDrawerManager 的类来抽象出处理抽屉布局的代码。要构造这个对象,我需要传入并保留对 Activity 的引用。我使用此活动引用来调用 findViewById(),并且在创建列表适配器等时也将活动用作上下文。我在活动中保留对 NavDrawerManager 对象的引用,以便我可以对抽屉布局执行回调和操作.

更新:根据 Xaver 的建议,我切换到将 FragmentActivity 扩展为 NavDrawerActivity。超类处理导航抽屉代码,而不是在我的活动中使用 NavDrawerManager 对象。请参阅更新的代码。

问题:

一切正常,直到我改变方向。更改方向后,我的 NavDrawerActivity 似乎仍然引用旧布局。我这样说是因为我在 DrawerLayout 中有一个进度条和几个按钮,我正在尝试更新它们,但它们并不反映更新。

当我调用 Activity.findViewById() 来获取进度条并使其可见时,它不会显示任何更改,也不会显示对按钮的任何更改。注意:Activity.findViewById() 不返回 null。但是,如果我在更改方向之前执行相同的操作,进度条和按钮会显示相应的更改。

代码:

导航抽屉活动:

public class NavDrawerActivity extends LowProfileFragmentActivity {
    private DrawerLayout mDrawerLayout;
    private ActionBarDrawerToggle mDrawerToggle;

    private ExpandableListView mDrawerList;
    private NavDrawerAdapter mListAdapter;

    @Override
    protected void onPostCreate(Bundle savedInstanceState) {
        super.onPostCreate(savedInstanceState);
        mDrawerToggle.syncState();
    }

    @Override
    public boolean onOptionsItemSelected(MenuItem item) {
        mDrawerToggle.onOptionsItemSelected(item);
        return super.onOptionsItemSelected(item);
    }

    //Called via subclass after setContentView    
    protected void setupNavDrawer() {
        initDrawerLayout();
        initDrawerList();
        setButton1();
        setButton2();
    }

    private void initDrawerLayout() {
        mDrawerLayout = (DrawerLayout) findViewById(R.id.drawer_layout);
        mDrawerToggle = getActionBarDrawerToggle();
        mDrawerLayout.setDrawerListener(mDrawerToggle);
    }

    private void initDrawerList() {
        mDrawerList = (ExpandableListView) mDrawerLayout
                .findViewById(android.R.id.list);

        mListAdapter = getNavDrawerAdapter();
    }

    private ActionBarDrawerToggle getActionBarDrawerToggle() {
        return new ActionBarDrawerToggle(this, mDrawerLayout,
                R.drawable.ic_navigation_drawer, 0, 0) {

            public void onDrawerClosed(View view) {

            }

            public void onDrawerOpened(View view) {

            }
        };
    }


    public void setDrawerLoading(boolean loading) {
        ProgressBar progressBar = (ProgressBar) mDrawerLayout
                .findViewById(R.id.progress_bar);
        Button button1 = (Button) this.findViewById(R.id.button1);
        Button button2 = (Button) this
                .findViewById(R.id.button2);

        /* None of the below changes appear after changing orientation */
        if (loading) {
            button1.setEnabled(false);
            button2.setEnabled(false);
            progressBar.setVisibility(View.VISIBLE);
        } else {
            progressBar.setVisibility(View.GONE);
            button1.setEnabled(true);
            button2.setEnabled(true);
        }
    }

    private void setButton1() {
        Button button1 = (Button) this.findViewById(R.id.button1);
        button1.setOnClickListener(new OnClickListener() {

            @Override
            public void onClick(View v) {
                //do something
            }

        });
    }

    private void setButton2() {
        Button button2 = (Button) this
                .findViewById(R.id.button2);
        button2.setOnClickListener(new OnClickListener() {

            @Override
            public void onClick(View v) {
                //do something
            }

        });
    }
}

示例活动:

public class ExampleActivity extends NavDrawerActivity {
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        initUI();
    }

    private void initUI() {
        setContentView(R.layout.activity_sift);

        // Call the superclass method to set up the nav drawer
        setupNavDrawer();
    }
}

AndroidManifest.xml:

<activity
    android:name="com.example.app.ExampleActivity"
    android:label="@string/app_name"
    android:launchMode="singleTop"
    android:theme="@style/AppTheme" >
    <intent-filter>
        <action android:name="android.intent.action.MAIN" />
            <category android:name="android.intent.category.LAUNCHER" />
    </intent-filter>
</activity>
4

1 回答 1

1

如果您实施 aNavDrawerActivity而不是 a会不会更容易NavDrawerManager?一般来说,应该避免这样的事情。UI 应仅由他们Fragments自己处理Activities

我理解您为什么要创建这样的NavDrawerManager,只是为了使整个事物可重用并且大量抽象很棒,但是当它过度使用时,它可能会成为一个问题。不要过分关注完全抽象并符合 OOP 标准的所有内容。Activity同时拥有NavigationDrawer和隐藏通知栏没有任何问题。如果你已经有一个Activity实现这两件事之一的,你应该扩展它。您唯一的其他选择是在一秒钟内重新实现相同的行为Activity,这几乎会破坏抽象的目的。


编辑:我已经重构和改进了你的代码,你真的不应该有这么多不同的方法,所有的设置都属于onCreate(),你为每个步骤编写额外的方法只是引入了额外错误的可能性。尝试这个:

public abstract class NavDrawerActivity extends LowProfileFragmentActivity {

    private ActionBarDrawerToggle drawerToggle;
    private DrawerLayout drawerLayout;
    private ProgressBar progressBar;
    private Button button1;
    private Button button2;

    private ExpandableListView drawerList;
    private NavDrawerAdapter drawerListAdapter;

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

        this.drawerLayout = (DrawerLayout) findViewById(R.id.drawer_layout);
        this.drawerToggle = new ActionBarDrawerToggle(this, this.drawerLayout, R.drawable.icon_drawer, R.string.drawer_open, R.string.drawer_close);
        this.drawerLayout.setDrawerListener(this.drawerToggle);

        this.drawerList = (ExpandableListView) this.drawerLayout.findViewById(android.R.id.list);
        this.progressBar = (ProgressBar) this.drawerLayout.findViewById(R.id.progress_bar);

        this.button1 = (Button) findViewById(R.id.button1);
        this.button1.setOnClickListener(new View.OnClickListener() {

            @Override
            public void onClick(View v) {
                //do something
            }
        });

        this.button2 = (Button) findViewById(R.id.button2);
        this.button2.setOnClickListener(new View.OnClickListener() {

            @Override
            public void onClick(View v) {
                //do something
            }
        });

        this.drawerListAdapter = new NavDrawerAdapter(...);
        this.drawerList.setAdapter(this.drawerListAdapter);
    }

    protected abstract int getLayout();

    @Override
    protected void onPostCreate(Bundle savedInstanceState) {
        super.onPostCreate(savedInstanceState);
        this.drawerToggle.syncState();
    }

    @Override
    public void onConfigurationChanged(Configuration newConfig) {
        super.onConfigurationChanged(newConfig);
        this.drawerToggle.onConfigurationChanged(newConfig);
    }

    @Override
    public boolean onOptionsItemSelected(MenuItem item) {
        return this.drawerToggle.onOptionsItemSelected(item) || super.onOptionsItemSelected(item);
    }

    public void setDrawerLoading(boolean loading) {
        this.button1.setEnabled(!loading);
        this.button2.setEnabled(!loading);
        this.progressBar.setVisibility(loading ? View.VISIBLE : View.GONE);
    }
}

正如你所看到的,我使用了一个抽象方法getLayout()来将正确的布局传递给setContentView(). getLayout()必须由所有Activities扩展NavDrawerActivitythetherfore 的人来实现,从而使 subActivities控制使用的布局。

因此,您ExampleActivity应该如下所示:

public class ExampleActivity extends NavDrawerActivity {

    @Override
    protected int getLayout() {
        return R.layout.activity_sift;
    }
}

我测试了一切,它对我有用。

于 2014-06-24T09:49:39.443 回答