android: getFragmentManager() возвращает ноль
Редактировать: проблема в том, что фрагмент отделяется от действия. Однако я не знаю, как это предотвратить.
У меня есть приложение, которое использует панель вкладок. При нажатии на одну из вкладок она загружает список тем в поле под панелью вкладок. Одна из тем - "Направления". После нажатия этой кнопки пространство теперь заменяется макетом, который имеет еще одну панель вверху (ниже панели вкладок), которая включает кнопку "Назад" и кнопку "Получить маршрут". Ниже этой панели находится карта местоположения Google. Если я нажму "Назад", я вернусь к списку, как и ожидалось. Затем, если я нажму "Направления" в списке, я вернусь к карте, как и ожидалось. Затем, если я снова нажму "Назад" вместо того, чтобы снова возвращаться к списку, это приведет к сбою приложения. Он показывает исключение NullPointerException, которое указывает на строку FragmentTransaction ft = getFragmentManager().beginTransaction();
внутри setOnClickListener кнопки "Назад". Не уверен, почему он теряет след фрагмента менеджера.
Вот файл для вкладки, который изначально загружает список:
package org.childrensmuseum.visittcmindy;
import java.util.ArrayList;
import android.content.Context;
import android.graphics.drawable.GradientDrawable;
import android.graphics.drawable.GradientDrawable.Orientation;
import android.os.Bundle;
import android.support.v4.app.Fragment;
import android.support.v4.app.FragmentTransaction;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.AdapterView;
import android.widget.AdapterView.OnItemClickListener;
import android.widget.ArrayAdapter;
import android.widget.ImageView;
import android.widget.ListView;
import android.widget.TextView;
public class InfoTab extends Fragment {
private ArrayList<InfoPage> pages;
private TCMSQLiteHelper sqliteHelper;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
sqliteHelper = new TCMSQLiteHelper(this.getActivity());
pages = sqliteHelper.getAllPages();
InfoPage directions = new InfoPage();
directions.setPage_id(101);
directions.setTitle("Directions");
InfoPage calendar = new InfoPage();
calendar.setPage_id(102);
calendar.setTitle("Calendar");
pages.add(1, directions);
pages.add(2, calendar);
}
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState) {
View v = inflater.inflate(R.layout.infolist, container, false);
ListView infoList = (ListView)v.findViewById(R.id.infolistview);
int[] colors = {0xFFCFE8FF,0xFFCFE8FF};
infoList.setDivider(new GradientDrawable(Orientation.RIGHT_LEFT, colors));
infoList.setDividerHeight(2);
ArrayList<String> valueList = new ArrayList<String>();
for(InfoPage page : pages){
valueList.add(page.getTitle());
}
String[] values = new String[valueList.size()];
values = valueList.toArray(values);
infoList.setAdapter(new InfoAdapter(getActivity(), values));
infoList.setOnItemClickListener(new OnItemClickListener(){
@Override
public void onItemClick(AdapterView<?> parent, View v, int p,
long id) {
long page_id = pages.get(p).getPage_id();
// If Directions
if(page_id==101){
DirectionsView directions = new DirectionsView();
FragmentTransaction ft = getFragmentManager().beginTransaction();
ft.replace(R.id.realtabcontent, directions);
ft.commit();
} else {
PageDetails details = PageDetails.newInstance(pages.get(p).getPage_id());
FragmentTransaction ft = getFragmentManager().beginTransaction();
ft.replace(R.id.realtabcontent, details);
ft.commit();
}
}
});
return v;
}
private class InfoAdapter extends ArrayAdapter<String> {
public InfoAdapter(Context context, String[] values) {
super(context, R.layout.explorecell, values);
// TODO Auto-generated constructor stub
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
LayoutInflater inflater = (LayoutInflater) getActivity().getSystemService(Context.LAYOUT_INFLATER_SERVICE);
View row = inflater.inflate(R.layout.infocell,parent,false);
InfoPage page = pages.get(position);
if(position % 2 != 0){
row.setBackgroundColor(getResources().getColor(R.color.listblue));
} else {
row.setBackgroundColor(getResources().getColor(R.color.white));
}
String page_icon = "page_" + page.getPage_id();
ImageView iv = (ImageView)row.findViewById(R.id.infoImage);
TextView tv = (TextView)row.findViewById(R.id.infoText);
String title = page.getTitle();
String[] titleParts = title.split("[ ]");
String newTitle = "";
for(int i=0;i<titleParts.length;i++){
titleParts[i] = titleParts[i].substring(0,1).toUpperCase() + titleParts[i].substring(1);
newTitle += titleParts[i];
if(i<titleParts.length-1){
newTitle += " ";
}
}
iv.setImageResource(getActivity().getResources().getIdentifier(page_icon, "drawable","org.childrensmuseum.visittcmindy"));
tv.setText(newTitle);
return row;
}
}
@Override
public void onDestroy() {
super.onDestroy();
if(sqliteHelper != null) {
sqliteHelper.close();
}
}
}
Вот файл для DirectionsView, который вызывается при нажатии на этот элемент в списке:
package org.childrensmuseum.visittcmindy;
import java.io.UnsupportedEncodingException;
import java.net.URLEncoder;
import java.util.List;
import android.app.Dialog;
import android.content.Intent;
import android.location.Address;
import android.location.Geocoder;
import android.location.Location;
import android.net.Uri;
import android.os.Bundle;
import android.support.v4.app.Fragment;
import android.support.v4.app.FragmentTransaction;
import android.util.Log;
import android.view.InflateException;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.widget.EditText;
import com.google.android.gms.common.ConnectionResult;
import com.google.android.gms.common.GooglePlayServicesClient;
import com.google.android.gms.location.LocationClient;
import com.google.android.gms.maps.GoogleMap;
import com.google.android.gms.maps.SupportMapFragment;
import com.google.android.gms.maps.model.LatLng;
import com.google.android.gms.maps.model.MarkerOptions;
public class DirectionsView extends Fragment implements
GooglePlayServicesClient.ConnectionCallbacks,
GooglePlayServicesClient.OnConnectionFailedListener {
private static View v;
private MainApplication main;
private LocationClient mLocationClient;
@Override
public void onCreate(Bundle savedInstanceState) {
// TODO Auto-generated method stub
super.onCreate(savedInstanceState);
main = MainApplication.getInstance();
mLocationClient = new LocationClient(getActivity(), this, this);
}
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState) {
if(v != null){
ViewGroup parent = (ViewGroup) v.getParent();
if(parent != null){
parent.removeView(v);
}
}
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 {
Log.d("DEBUG","2");
String museumAddress = "The Children's Museum of Indianapolis, 3000 North Meridian St, Indianapolis, IN 46208";
address = coder.getFromLocationName(museumAddress, 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) {
Log.d("DEBUG","Error Adding Marker: "+e.getLocalizedMessage());
}
Button backButton = (Button)v.findViewById(R.id.backtoexhibits);
backButton.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
FragmentTransaction ft = getFragmentManager().beginTransaction();
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 */
e.printStackTrace();
}
return v;
}
@Override
public void onConnectionFailed(ConnectionResult result) {
// TODO Auto-generated method stub
}
@Override
public void onConnected(Bundle connectionHint) {
// TODO Auto-generated method stub
}
@Override
public void onDisconnected() {
// TODO Auto-generated method stub
}
private void openDirectionsDialog() throws UnsupportedEncodingException{
final Dialog dialog = new Dialog(getActivity());
dialog.setContentView(R.layout.getdirections);
dialog.setTitle("Get Directions");
Button useAddressBtn = (Button)dialog.findViewById(R.id.useAddress);
Button useLocationBtn = (Button)dialog.findViewById(R.id.useLocationBtn);
final EditText addressText = (EditText)dialog.findViewById(R.id.addressText);
final String museumAddress = URLEncoder.encode("The Children's Museum of Indianapolis, 3000 North Meridian St, Indianapolis, IN 46208","UTF-8");
useAddressBtn.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
String address;
try {
address = URLEncoder.encode(addressText.getText().toString(),"UTF-8");
String url = "http://maps.google.com/maps?saddr="+address+"&daddr="+museumAddress;
Intent intent = new Intent(android.content.Intent.ACTION_VIEW, Uri.parse(url));
intent.setClassName("com.google.android.apps.maps", "com.google.android.maps.MapsActivity");
startActivity(intent);
} catch (UnsupportedEncodingException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
});
useLocationBtn.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
Location mCurrentLocation = mLocationClient.getLastLocation();
double lat = mCurrentLocation.getLatitude();
double lng = mCurrentLocation.getLongitude();
String url = "http://maps.google.com/maps?saddr="+lat+","+lng+"&daddr="+museumAddress;
Intent intent = new Intent(android.content.Intent.ACTION_VIEW, Uri.parse(url));
intent.setClassName("com.google.android.apps.maps", "com.google.android.maps.MapsActivity");
startActivity(intent);
}
});
dialog.show();
}
}
Редактировать:
Вот трассировка стека
12-11 14:07:03.547: E/AndroidRuntime(1846): FATAL EXCEPTION: main
12-11 14:07:03.547: E/AndroidRuntime(1846): Process: org.childrensmuseum.visittcmindy, PID: 1846
12-11 14:07:03.547: E/AndroidRuntime(1846): java.lang.NullPointerException
12-11 14:07:03.547: E/AndroidRuntime(1846): at org.childrensmuseum.visittcmindy.DirectionsView$1.onClick(DirectionsView.java:80)
12-11 14:07:03.547: E/AndroidRuntime(1846): at android.view.View.performClick(View.java:4424)
12-11 14:07:03.547: E/AndroidRuntime(1846): at android.view.View$PerformClick.run(View.java:18383)
12-11 14:07:03.547: E/AndroidRuntime(1846): at android.os.Handler.handleCallback(Handler.java:733)
12-11 14:07:03.547: E/AndroidRuntime(1846): at android.os.Handler.dispatchMessage(Handler.java:95)
12-11 14:07:03.547: E/AndroidRuntime(1846): at android.os.Looper.loop(Looper.java:137)
12-11 14:07:03.547: E/AndroidRuntime(1846): at android.app.ActivityThread.main(ActivityThread.java:4998)
12-11 14:07:03.547: E/AndroidRuntime(1846): at java.lang.reflect.Method.invokeNative(Native Method)
12-11 14:07:03.547: E/AndroidRuntime(1846): at java.lang.reflect.Method.invoke(Method.java:515)
12-11 14:07:03.547: E/AndroidRuntime(1846): at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:777)
12-11 14:07:03.547: E/AndroidRuntime(1846): at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:593)
12-11 14:07:03.547: E/AndroidRuntime(1846): at dalvik.system.NativeStart.main(Native Method)
Изменить 2:
В соответствии с предложениями я попытался заменить
FragmentTransaction ft = getFragmentManager().beginTransaction();
с
FragmentTransaction ft = getActivity().getSupportFragmentManager().beginTransaction();
Там не было никаких изменений в ошибке.
2 ответа
Я рекомендую это: http://developer.android.com/training/implementing-navigation/lateral.html
Это руководство по использованию viewPager и gmentPagerAdapter с вкладками навигации, так что вам не нужно вручную настраивать onClickListener для изменения фрагментов. Это должно предотвратить отрыв фрагментов.
Вы перепутали фрагмент поддержки и фрагмент. Вы должны использовать библиотеку ActionBarSherlock для этого. Так что вам нужно продлить SherlockFragment
и не Fragment
,
После этого вам нужно использовать редактирование, которое вы дали:
FragmentTransaction ft = getActivity().getSupportFragmentManager().beginTransaction();
Это должно заставить его работать.