web-dev-qa-db-ja.com

方向が異なるActionBarのタブにフラグメントを割り当てる

3つのフラグメントと1つのアクティビティがあります。 ActionBarのタブを有効にして、Fragmentを3つのタブのそれぞれに割り当てます。どうすれば正しく接続できますか?

元の投稿

Google I/Oアプリをガイドとして使用して開発しているアプリがあります。 ActionBarにタブを実装しました。タブレットの向きが変わるまでは動作しているようです。たとえば、3つのタブすべてにFragmentがあります。私はそれらをうまく切り替えることができますが、方向を変更すると、最後に表示していたFragmentは何も表示されたままですが、タブをクリックしてもビューが変更されることはありません。予想通り、元の向きに戻っても「修正」されません。

状態の保存と復元を検討しましたが、それらがどのように役立つかわかりません。

[〜#〜]編集[〜#〜]

モジュールレベル:

Fragment mFragmentA = (Fragment) new AFragmentTab();
Fragment mFragmentB = (Fragment) new BFragmentTab();
Fragment mFragmentC = (Fragment) new CFragmentTab();

アクティビティのonCreateに次のようなものがあります:

ActionBar actionBar = getActionBar();
actionBar.setNavigationMode(ActionBar.NAVIGATION_MODE_TABS);
actionBar.setDisplayShowTitleEnabled(false);

ActionBar.Tab tabA = actionBar.newTab().setText("text a");
ActionBar.Tab tabB = actionBar.newTab().setText("text b");
ActionBar.Tab tabC = actionBar.newTab().setText("text c");

tabA.setTabListener(this);
tabB.setTabListener(this);
tabC.setTabListener(this);

actionBar.addTab(tabA);
actionBar.addTab(tabB);
actionBar.addTab(tabC);

そして、次のようなTabListener

[〜#〜] edit [〜#〜]これは削除されます

class MyTabListener implements ActionBar.TabListener {
    private Fragment mFragment;

    // Called to create an instance of the listener when adding a new tab
    public MyTabListener(Fragment fragment) {
        mFragment = fragment;
    }

    public void onTabSelected(Tab tab, FragmentTransaction ft) {
        ft.add(R.id.fragment_content, mFragment, null);
    }

    public void onTabUnselected(Tab tab, FragmentTransaction ft) {
        ft.remove(mFragment);
    }

    public void onTabReselected(Tab tab, FragmentTransaction ft) {
        // do nothing
    }
}

[〜#〜]編集[〜#〜]

TabListenerを移動しました。個別のクラスではなく、TabListenerActivityに実装します。次に、SelectedおよびUnselectedメソッドについて、次のようなものがあります。

public void onTabSelected(Tab tab, FragmentTransaction ft) {
    switch (tab.getPosition()) {
    case 0:
        ft.add(R.id.fragment_content, mFragmentA, null);
        break;
    case 1:
        ft.add(R.id.fragment_content, mFragmentB, null);
        break;
    case 2:
        ft.add(R.id.fragment_content, mFragmentC, null);
        break;
    }
}

public void onTabUnselected(Tab tab, FragmentTransaction ft) {
    switch (tab.getPosition()) {
    case 0:
        ft.remove(mFragmentA);
        break;
    case 1:
        ft.remove(mFragmentB);
        break;
    case 2:
        ft.remove(mFragmentC);
        break;
    }
}

まだ同じことをしています。何が起こっているのか本当にわかりません。

25
Metallicraft

私の答えはここにあるようです: http://developer.Android.com/resources/samples/ApiDemos/src/com/example/Android/apis/app/FragmentTabs.html

/*
 * Copyright (C) 2011 The Android Open Source Project
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.Apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package com.example.Android.apis.app;

import com.example.Android.apis.R;


import Android.app.ActionBar;
import Android.app.ActionBar.Tab;
import Android.app.Activity;
import Android.app.Fragment;
import Android.app.FragmentTransaction;
import Android.os.Bundle;
import Android.widget.Toast;

/**
 * This demonstrates the use of action bar tabs and how they interact
 * with other action bar features.
 */
public class FragmentTabs extends Activity {
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);

        final ActionBar bar = getActionBar();
        bar.setNavigationMode(ActionBar.NAVIGATION_MODE_TABS);
        bar.setDisplayOptions(0, ActionBar.DISPLAY_SHOW_TITLE);

        bar.addTab(bar.newTab()
                .setText("Simple")
                .setTabListener(new TabListener<FragmentStack.CountingFragment>(
                        this, "simple", FragmentStack.CountingFragment.class)));
        bar.addTab(bar.newTab()
                .setText("Contacts")
                .setTabListener(new TabListener<LoaderCursor.CursorLoaderListFragment>(
                        this, "contacts", LoaderCursor.CursorLoaderListFragment.class)));
        bar.addTab(bar.newTab()
                .setText("Apps")
                .setTabListener(new TabListener<LoaderCustom.AppListFragment>(
                        this, "apps", LoaderCustom.AppListFragment.class)));
        bar.addTab(bar.newTab()
                .setText("Throttle")
                .setTabListener(new TabListener<LoaderThrottle.ThrottledLoaderListFragment>(
                        this, "throttle", LoaderThrottle.ThrottledLoaderListFragment.class)));

        if (savedInstanceState != null) {
            bar.setSelectedNavigationItem(savedInstanceState.getInt("tab", 0));
        }
    }

    @Override
    protected void onSaveInstanceState(Bundle outState) {
        super.onSaveInstanceState(outState);
        outState.putInt("tab", getActionBar().getSelectedNavigationIndex());
    }

    public static class TabListener<T extends Fragment> implements ActionBar.TabListener {
        private final Activity mActivity;
        private final String mTag;
        private final Class<T> mClass;
        private final Bundle mArgs;
        private Fragment mFragment;

        public TabListener(Activity activity, String tag, Class<T> clz) {
            this(activity, tag, clz, null);
        }

        public TabListener(Activity activity, String tag, Class<T> clz, Bundle args) {
            mActivity = activity;
            mTag = tag;
            mClass = clz;
            mArgs = args;

            // Check to see if we already have a fragment for this tab, probably
            // from a previously saved state.  If so, deactivate it, because our
            // initial state is that a tab isn't shown.
            mFragment = mActivity.getFragmentManager().findFragmentByTag(mTag);
            if (mFragment != null && !mFragment.isDetached()) {
                FragmentTransaction ft = mActivity.getFragmentManager().beginTransaction();
                ft.hide(mFragment);
                ft.commit();
            }
        }

        public void onTabSelected(Tab tab, FragmentTransaction ft) {
            if (mFragment == null) {
                mFragment = Fragment.instantiate(mActivity, mClass.getName(), mArgs);
                ft.add(Android.R.id.content, mFragment, mTag);
            } else {
                ft.show(mFragment);
            }
        }

        public void onTabUnselected(Tab tab, FragmentTransaction ft) {
            if (mFragment != null) {
                ft.hide(mFragment);
            }
        }

        public void onTabReselected(Tab tab, FragmentTransaction ft) {
            Toast.makeText(mActivity, "Reselected!", Toast.LENGTH_SHORT).show();
        }
    }
}
22
Metallicraft

@Metallicraftによって提供されたリンクは無効になっているので、この投稿にまだ遭遇している人のためにここにいくつかの助けがあります。タブの実装のヘルプについては here を参照してください。

元のサンプルを表示/読み取るには、Android開発環境をインストールしている場合、次のフォルダーに移動できます:Android-sdk\samples\Android-14\ApiDemos\src\com\example\Android\apis\app\FragmentTabs.Java

また、方向変更後にフラグメントが重複する問題がある場合は、渡されたsavedinstancestateに適切に関連して変更します。 google tutorial からのコピーペーストは、フラグメントの重複の問題とそれを回避する方法を説明しています。

import Android.os.Bundle;
import Android.support.v4.app.FragmentActivity;

public class MainActivity extends FragmentActivity {
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.news_articles);

        // Check that the activity is using the layout version with
        // the fragment_container FrameLayout
        if (findViewById(R.id.fragment_container) != null) {

            // However, if we're being restored from a previous state,
            // then we don't need to do anything and should return or else
            // we could end up with overlapping fragments.
            if (savedInstanceState != null) {
                return;
            }

            // Create an instance of ExampleFragment
            HeadlinesFragment firstFragment = new HeadlinesFragment();

            // In case this activity was started with special instructions from an Intent,
            // pass the Intent's extras to the fragment as arguments
            firstFragment.setArguments(getIntent().getExtras());

            // Add the fragment to the 'fragment_container' FrameLayout
            getSupportFragmentManager().beginTransaction()
                    .add(R.id.fragment_container, firstFragment).commit();
        }
    }
}
7
Warpzit

新しいAndroidサンプルプロジェクトを作成し、Support4Demosサンプルを選択します。FragmentTabsサンプルがそこにあります。

2
azevik

私は多かれ少なかれ同じ問題を抱えていましたが、上記の解決策は私の状況ではうまくいかなかったようです。最終的に私は次の解決策を見つけました:

    @Override
    public void onTabSelected(Tab tab, FragmentTransaction ft) {
        if (mFragment == null) {
            mFragment = Fragment.instantiate(mActivity, mClass.getName());
            ft.replace(Android.R.id.content, mFragment, mTag); // Use replace iso add
        }
        else {
            ft.attach(mFragment);
        }
    }
1
Boeboe
        @Override
        public void onTabSelected(ActionBar.Tab tab, Android.support.v4.app.FragmentTransaction fragmentTransaction) {
            // Check if the fragment is already initialized

            if (mFragment == null) {
                // If not, instantiate and add it to the activity
               // Toast.makeText(getApplicationContext(),"TAb "+tab.getPosition(),Toast.LENGTH_LONG).show();
                if(tab.getPosition() == 1) {
                   mFragment = new HolderFragment();
                    fragmentTransaction.add(Android.R.id.content, mFragment, mTag);
                }
                else if(tab.getPosition() == 0)
                {
                    mFragment = new FragmentKnowledge();
                    fragmentTransaction.add(Android.R.id.c`enter code here`ontent, mFragment, mTag);

                }
                /*else
                {

                     mFragment = Fragment.instantiate(mActivity, mClass.getName());
                    fragmentTransaction.add(Android.R.id.content, mFragment, mTag);
                }
*/
            } else {
                // If it exists, simply attach it in order to show it
                fragmentTransaction.attach(mFragment);
            }
        }
0
Aye Su Naing

これは私にとってはうまくいきました!タブを変更するたびにアクティビティが重複するという問題がありました:

        mFragment = mActivity.getFragmentManager().findFragmentByTag(mTag);
        if (mFragment != null && !mFragment.isDetached()) {
            FragmentTransaction ft = mActivity.getFragmentManager().beginTransaction();
            ft.detach(mFragment);
            ft.commit();

再度、感謝します!

0
saman0suke