3

我有一个顶部有标签栏的应用程序。单击其中一个选项卡时,它会加载一个 ListView。然后,当单击列表视图项之一并加载详细信息页面时。详细信息页面在选项卡栏下方的顶部有另一个工具栏,带有一个返回按钮可以返回到列表。列表选项之一是“方向”。这将加载一个视图,而不是正常的详细信息页面,该视图在顶部包含相同的工具栏(包括获取方向按钮),其余部分是谷歌地图,以带有标记的位置为中心。这最初效果很好。如果我单击后退按钮,它会很好地进入列表,我可以返回方向。但是,有两件事打破了这一点。

  1. 我在方向页面上。我单击另一个选项卡。然后我单击返回上一个选项卡。现在列表再次出现(应该如此)。然后我点击“方向”按钮。我没有显示工具栏和地图,而是在标签栏下方看到一个空白屏幕。

  2. 同样,我在方向页面上。我单击返回返回列表。然后我单击另一个选项卡。然后我单击返回上一个选项卡。我再次点击“路线”。应用程序崩溃并抛出 NullPointerException。例外是在我尝试通过它的 id 检索地图的那一行。

这是在点击路线列表项并加载路线页面时调用的代码:

    try {
            mLocationClient.connect();
            v = inflater.inflate(R.layout.directions, container, false);
            GoogleMap mMap;
            mMap = ((SupportMapFragment) getFragmentManager().findFragmentById(R.id.directionsmap)).getMap();
            Geocoder coder = new Geocoder(getActivity());
            List<Address> address;

            try {
                String myAddress = "1 Monument Cir Indianapolis, IN 46204";
                address = coder.getFromLocationName(myAddress, 1);
                Address museumLocation = address.get(0);
                mMap.addMarker(new MarkerOptions()
                        .position(new LatLng(museumLocation.getLatitude(), museumLocation.getLongitude()))
                        .title(getResources().getString(R.string.museumTitle)));
            } catch(Exception e) {
                //
            }
            Button backButton = (Button)v.findViewById(R.id.backtoexhibits);
            backButton.setOnClickListener(new OnClickListener() {
                @Override
                public void onClick(View v) {
                    //Fragment fragment = (getFragmentManager().findFragmentById(R.id.directionsmap));   
                    FragmentTransaction ft = getActivity().getSupportFragmentManager().beginTransaction();
                    //ft.remove(fragment);

                    InfoTab infoTab = new InfoTab();
                    ft.replace(R.id.realtabcontent, infoTab);
                    ft.commit();
                }
            });

            Button directionsButton = (Button)v.findViewById(R.id.getdirections);
            directionsButton.setOnClickListener(new OnClickListener() {

                @Override
                public void onClick(View v) {
                    try {
                        openDirectionsDialog();
                    } catch (UnsupportedEncodingException e) {
                        // TODO Auto-generated catch block
                        e.printStackTrace();
                    }
                }
            });
        } catch (InflateException e){
            /* map is already there, just return view as it is */
            Log.d("DEBUG",e.getLocalizedMessage());
        }

这是我的方向片段的 xml:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:map="http://schemas.android.com/apk/res-auto"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical" >
<LinearLayout
    android:id="@+id/directionsheader"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:background="@color/tcmblue"
    android:orientation="horizontal" >
    <Button
        android:id="@+id/backtoexhibits"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="@string/backtext"
        android:layout_margin="5dp"
        android:textColor="@color/white"
        android:background="@drawable/buttonselector"
        android:paddingLeft="10dp"
        android:paddingRight="10dp"
        android:paddingTop="5dp"
        android:paddingBottom="5dp" />
    <Button
        android:id="@+id/getdirections"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="@string/getdirections"
        android:layout_margin="5dp"
        android:textColor="@color/white"
        android:background="@drawable/buttonselector"
        android:paddingLeft="10dp"
        android:paddingRight="10dp"
        android:paddingTop="5dp"
        android:paddingBottom="5dp" />

</LinearLayout>
<fragment 
    class="com.google.android.gms.maps.SupportMapFragment"
    android:id="@+id/directionsmap"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    map:cameraTargetLat="39.810166"
    map:cameraTargetLng="-86.156708"
    map:cameraZoom="15"
    map:mapType="normal" />

</LinearLayout>

编辑

这是我的标签活动的代码。

import android.content.Intent;
import android.os.Bundle;
import android.support.v4.app.FragmentActivity;
import android.support.v4.app.FragmentTabHost;
import android.support.v4.app.FragmentTransaction;
import android.widget.TabHost.OnTabChangeListener;
import android.widget.TabWidget;
import android.widget.TextView;

public class TabsActivity extends FragmentActivity {
    private FragmentTabHost mTabHost;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.tabs);
        mTabHost = (FragmentTabHost)findViewById(android.R.id.tabhost);
        mTabHost.setup(this, getSupportFragmentManager(), R.id.realtabcontent);

        mTabHost.addTab(mTabHost.newTabSpec("Home").setIndicator("Home", getResources().getDrawable(R.drawable.hometab)),HomeTab.class, null);
        mTabHost.addTab(mTabHost.newTabSpec("Explore").setIndicator("Explore", getResources().getDrawable(R.drawable.exploretab)),ExploreTab.class, null);
        mTabHost.addTab(mTabHost.newTabSpec("Info").setIndicator("Info", getResources().getDrawable(R.drawable.infotab)),InfoTab.class, null);
        mTabHost.addTab(mTabHost.newTabSpec("Social").setIndicator("Social", getResources().getDrawable(R.drawable.socialtab)),SocialTab.class, null);
        mTabHost.addTab(mTabHost.newTabSpec("Contact").setIndicator("Contact", getResources().getDrawable(R.drawable.contacttab)),ContactTab.class, null);

        TabWidget tabWidget = mTabHost.getTabWidget();
        tabWidget.setStripEnabled(false);
        for(int i=0; i < tabWidget.getChildCount(); i++){
            tabWidget.getChildAt(i).setBackgroundResource(R.drawable.tab_bg);
            TextView tv = (TextView)tabWidget.getChildAt(i).findViewById(android.R.id.title);
            tv.setTextColor(this.getResources().getColor(R.color.white));
        }

        mTabHost.setOnTabChangedListener(new OnTabChangeListener() {

            @Override
            public void onTabChanged(String tabId) {
                FragmentTransaction ft = getSupportFragmentManager().beginTransaction();
                if(tabId=="Home"){
                    finish();
                } else if(tabId=="Explore") {
                    ExploreTab exploreTab = new ExploreTab();
                    ft.replace(R.id.realtabcontent, exploreTab);
                } else if(tabId=="Info") {
                    InfoTab infoTab = new InfoTab();
                    ft.replace(R.id.realtabcontent, infoTab);
                } else if(tabId=="Social") {
                    SocialTab socialTab = new SocialTab();
                    ft.replace(R.id.realtabcontent, socialTab);
                } else if(tabId=="Contact") {
                    ContactTab contactTab = new ContactTab();
                    ft.replace(R.id.realtabcontent, contactTab);
                }
                ft.commit();
                TabWidget tw = mTabHost.getTabWidget();
                for(int i=0; i < tw.getChildCount(); i++){
                    TextView tabText = (TextView)tw.getChildAt(i).findViewById(android.R.id.title);
                    if(tabText.getText()==tabId){
                        tabText.setTextColor(TabsActivity.this.getResources().getColor(R.color.tcmgreen));
                    } else {
                        tabText.setTextColor(TabsActivity.this.getResources().getColor(R.color.white));
                    }

                }
            }
        });

        Intent intent = getIntent();
        String tag = intent.getStringExtra("tab");
        mTabHost.setCurrentTabByTag(tag);

    }

    }

编辑2:

我已经注释掉了后退按钮上删除片段的部分。该应用程序不再崩溃。但是,我可以去指示一次。之后,如果我回击或另一个选项卡,然后返回方向,我会得到一个空白屏幕,地图应该在哪里。以下是堆栈跟踪的内容:

12-09 16:29:56.056: W/System.err(16474): android.view.InflateException: Binary XML file line #39: Error inflating class fragment
12-09 16:29:56.066: W/System.err(16474):    at android.view.LayoutInflater.createViewFromTag(LayoutInflater.java:713)
12-09 16:29:56.066: W/System.err(16474):    at android.view.LayoutInflater.rInflate(LayoutInflater.java:755)
12-09 16:29:56.066: W/System.err(16474):    at android.view.LayoutInflater.inflate(LayoutInflater.java:492)
12-09 16:29:56.066: W/System.err(16474):    at android.view.LayoutInflater.inflate(LayoutInflater.java:397)
12-09 16:29:56.066: W/System.err(16474):    at org.childrensmuseum.visittcmindy.PageDetails.onCreateView(PageDetails.java:117)
12-09 16:29:56.066: W/System.err(16474):    at android.support.v4.app.Fragment.performCreateView(Fragment.java:1478)
12-09 16:29:56.066: W/System.err(16474):    at android.support.v4.app.FragmentManagerImpl.moveToState(FragmentManager.java:927)
12-09 16:29:56.066: W/System.err(16474):    at android.support.v4.app.FragmentManagerImpl.moveToState(FragmentManager.java:1104)
12-09 16:29:56.066: W/System.err(16474):    at android.support.v4.app.BackStackRecord.run(BackStackRecord.java:682)
12-09 16:29:56.066: W/System.err(16474):    at android.support.v4.app.FragmentManagerImpl.execPendingActions(FragmentManager.java:1460)
12-09 16:29:56.066: W/System.err(16474):    at android.support.v4.app.FragmentManagerImpl$1.run(FragmentManager.java:440)
12-09 16:29:56.066: W/System.err(16474):    at android.os.Handler.handleCallback(Handler.java:733)
12-09 16:29:56.066: W/System.err(16474):    at android.os.Handler.dispatchMessage(Handler.java:95)
12-09 16:29:56.066: W/System.err(16474):    at android.os.Looper.loop(Looper.java:137)
12-09 16:29:56.066: W/System.err(16474):    at android.app.ActivityThread.main(ActivityThread.java:4998)
12-09 16:29:56.066: W/System.err(16474):    at java.lang.reflect.Method.invokeNative(Native Method)
12-09 16:29:56.066: W/System.err(16474):    at java.lang.reflect.Method.invoke(Method.java:515)
12-09 16:29:56.066: W/System.err(16474):    at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:777)
12-09 16:29:56.066: W/System.err(16474):    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:593)
12-09 16:29:56.066: W/System.err(16474):    at dalvik.system.NativeStart.main(Native Method)
12-09 16:29:56.066: W/System.err(16474): Caused by: java.lang.IllegalArgumentException: Binary XML file line #39: Duplicate id 0x7f050058, tag null, or parent id 0x0 with another fragment for com.google.android.gms.maps.SupportMapFragment
12-09 16:29:56.066: W/System.err(16474):    at android.support.v4.app.FragmentActivity.onCreateView(FragmentActivity.java:290)
12-09 16:29:56.066: W/System.err(16474):    at android.view.LayoutInflater.createViewFromTag(LayoutInflater.java:685)

编辑 3

我现在把它修好了,这样你就可以在点击返回后返回地图。修复相当简单。您没有看到的是,在 try 语句之外,我的视图“v”是在 onCreateView 函数中声明的。我把它拿出来,并使它成为类中的私有静态变量。现在它不会丢失。我用这个线程来回答:Duplicate ID, tag null, or parent id with another fragment for com.google.android.gms.maps.MapFragment

一个新的问题出现了,如果你去地图,点击返回,然后返回到它,返回按钮会导致崩溃。它在创建 FragmentTransaction 时触发空指针异常。不过,我会为此启动一个新线程。

4

2 回答 2

2

这是我第一次尝试回答 SO 问题,所以如果我在这里偏离基地,我深表歉意!

在这种情况下,我并不完全熟悉您如何使用 FragmentTransaction,但我怀疑是对ft.remove(fragment);的调用 实际上正在中断后续调用以查找片段。

详细地说,按下后退按钮时,您似乎正在执行以下操作(从高级别):

  • 查找地图片段
  • 开始交易
  • 从片段管理器中删除地图片段(我认为这是问题所在
  • 交换片段内容(正如我所期望的那样)
  • 提交交易

如果您正在删除地图片段,并且您觉得确实需要删除它,那么您什么时候重新创建该片段并将其添加回管理器以供将来使用?

于 2013-12-07T06:11:39.577 回答
0

如上面编辑 3 所述。该修复需要将“v”的声明移到 onCreateView 方法之外。我把它作为一个私有静态变量放在类的顶部。灵感来自这个主题:Duplicate ID, tag null, or parent id with another fragment for com.google.android.gms.maps.MapFragment

出现了一个新问题,但它与地图无关。

于 2013-12-11T16:06:33.580 回答