Поддержка фрагментов для Mapsforge

Я знаю, что здесь есть какой-то патч для фрагментов для Mapsforge: https://code.google.com/p/mapsforge/issues/detail?id=177 Однако я не совсем уверен, как его использовать.

Это то, что я реализовал до сих пор:

public class TOfflineMapViewFragment  extends SherlockFragment{

    MapView myOpenMapView;
    //other codes here...

    public View onCreateView(LayoutInflater inflator, ViewGroup container, Bundle savedInstanceState) {

            view = inflator.inflate(R.layout.offline_map_activity, container, false);

            myOpenMapView = (MapView) view.findViewById(R.id.openmapview);
            myOpenMapView.setMapFile(new File(Environment.getExternalStorageDirectory().toString() + 
            "/offlineMap/singapore.map");

           return view;
    }

}

Это мой XML-файл:

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical" >
    <org.mapsforge.android.maps.MapView
        android:id="@+id/openmapview"
        android:layout_width="fill_parent"
        android:layout_height="fill_parent"/>
</LinearLayout>

Я получаю эту ошибку:

03-19 15:51:04.243: E/AndroidRuntime(10395): FATAL EXCEPTION: main

03-19 15:51:04.243: E/AndroidRuntime(10395): android.view.InflateException: Binary XML file line #6: Error inflating class org.mapsforge.android.maps.MapView
03-19 15:51:04.243: E/AndroidRuntime(10395):    at android.view.LayoutInflater.createView(LayoutInflater.java:606)
03-19 15:51:04.243: E/AndroidRuntime(10395):    at android.view.LayoutInflater.createViewFromTag(LayoutInflater.java:680)
03-19 15:51:04.243: E/AndroidRuntime(10395):    at android.view.LayoutInflater.rInflate(LayoutInflater.java:739)
03-19 15:51:04.243: E/AndroidRuntime(10395):    at android.view.LayoutInflater.inflate(LayoutInflater.java:489)
03-19 15:51:04.243: E/AndroidRuntime(10395):    at android.view.LayoutInflater.inflate(LayoutInflater.java:396)
03-19 15:51:04.243: E/AndroidRuntime(10395):    at com.fragments.TOfflineMapViewFragment.onCreateView(TOfflineMapViewFragment.java:76)

Что не так с этим?

2 ответа

Добавление

AndroidGraphicFactory.createInstance(getActivity().getApplication());

у меня в переопределенном onCreate() работал

Я создал шаблон фрагмента, который делает то же самое, что MapVeiwTemplate. Это расширяет Фрагмент вместо Действия.

import android.app.Fragment;
import android.content.Context;
import android.os.Bundle;
import android.os.Environment;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;

import org.mapsforge.core.model.LatLong;
import org.mapsforge.core.model.MapPosition;
import org.mapsforge.map.android.graphics.AndroidGraphicFactory;
import org.mapsforge.map.android.util.AndroidPreferences;
import org.mapsforge.map.android.view.MapView;
import org.mapsforge.map.datastore.MapDataStore;
import org.mapsforge.map.layer.cache.TileCache;
import org.mapsforge.map.layer.hills.HillsRenderConfig;
import org.mapsforge.map.model.MapViewPosition;
import org.mapsforge.map.model.common.PreferencesFacade;
import org.mapsforge.map.reader.MapFile;
import org.mapsforge.map.rendertheme.XmlRenderTheme;

import java.io.File;
import java.util.ArrayList;
import java.util.List;

/**
 * Created by a7med on 08.03.18.
 */
public abstract class MapFragmentTemplate extends Fragment
{
    // Abstract variables for displaying the map
    protected MapView mapView;
    protected PreferencesFacade preferencesFacade;
    protected List<TileCache> tileCaches = new ArrayList<TileCache>();

    // Constant variables for controlling the map
    private final static int MINZOOM = 5;
    private final static int MAXZOOM = 23;
    private final static int CURRENTZOOM = 12;

    // Default address in Norway ... Oslo is chosen as Default
    final LatLong DEFAULT_LOCATION = new LatLong(59.8939225, 10.7150775);

    /*
     * Abstract methods that must be implemented:
     *  - getLayoutId();
     *  - getMapViewId();
     *  - getMapFileName();
     *  - getRenderTheme();
     *  - createLayers();
     *  - createTileCaches();
     */
    /**
     * @return the layout to be used,
     */
    protected abstract int getLayoutId();

    /**
     * @return the id of the mapview inside the layout.
     */
    protected abstract int getMapViewId();

    /**
     * Gets the name of the map file.
     * The directory for the file is supplied by getMapFileDirectory()
     *
     * @return the map file name to be used
     */
    protected abstract String getMapFileName();

    /**
     * @return the rendertheme for this viewer
     */
    protected abstract XmlRenderTheme getRenderTheme();

    /**
     * Hook to create map layers. You will need to create at least one layer to
     * have something visible on the map.
     */
    protected abstract void createLayers();

    /**
     * Hook to create tile caches. For most map viewers you will need a tile cache
     * for the renderer. If you do not need tilecaches just provide an empty implementation.
     */
    protected abstract void createTileCaches();

    /*
        Methods for this class
     */
    /**
     * Hook to create controls, such as scale bars.
     * You can add more controls.
     */
    protected void createControls()
    {
        initializePosition(mapView.getModel().mapViewPosition);
        mapView.setCenter(DEFAULT_LOCATION);
        mapView.setZoomLevel((byte) CURRENTZOOM);
    }

    /**
     * The MaxTextWidthFactor determines how long a text may be before it is line broken. The
     * default setting should be good enough for most apps.
     *
     * @return the maximum text width factor for line breaking captions
     */
    protected float getMaxTextWidthFactor()
    {
        return 0.7f;
    }

    /**
     * @return the default starting zoom level if nothing is encoded in the map file.
     */
    protected byte getZoomLevelDefault()
    {
        return (byte) CURRENTZOOM;
    }

    /**
     * @return the minimum zoom level of the map view.
     */
    protected byte getZoomLevelMin()
    {
        return (byte) MINZOOM;
    }

    /**
     * @return the maximum zoom level of the map view.
     */
    protected byte getZoomLevelMax()
    {
        return (byte) MAXZOOM;
    }

    /**
     * Template method to create the map views.
     */
    protected void createMapViews(View v)
    {
        mapView = getMapView(v);
        mapView.getModel().init(this.preferencesFacade);
        mapView.setClickable(true);
        mapView.getMapScaleBar().setVisible(true);
        mapView.setBuiltInZoomControls(false);
        mapView.setZoomLevelMin((byte) MINZOOM);
        mapView.setZoomLevelMax((byte) MAXZOOM);
        mapView.setZoomLevel((byte) CURRENTZOOM);
        mapView.setCenter(new LatLong(DEFAULT_LOCATION.latitude, DEFAULT_LOCATION.longitude));
    }

    /**
     * Creates the shared preferences that are being used to store map view data over
     * activity restarts.
     */
    protected void createSharedPreferences()
    {
        this.preferencesFacade = new AndroidPreferences(getActivity().getSharedPreferences(getPersistableId(), Context.MODE_PRIVATE));
    }

    /**
     * Gets the default initial position of a map view if nothing is set in the map file. This
     * operation is used as a fallback only. Override this if you are not sure if your map file
     * will always have an initial position.
     *
     * @return the fallback initial position of the mapview.
     */
    protected MapPosition getDefaultInitialPosition()
    {
        return new MapPosition(DEFAULT_LOCATION, getZoomLevelDefault());
    }

    /**
     * Extracts the initial position from the map file, falling back onto the value supplied
     * by getDefaultInitialPosition if there is no initial position coded into the map file.
     * You will only need to override this method if you do not want the initial position extracted
     * from the map file.
     *
     * @return the initial position encoded in the map file or a fallback value.
     */
    protected MapPosition getInitialPosition()
    {
        final MapDataStore mapFile = getMapFile();

        if (mapFile.startPosition() != null)
        {
            Byte startZoomLevel = mapFile.startZoomLevel();
            if (startZoomLevel == null)
            {
                // it is actually possible to have no start zoom level in the file
                startZoomLevel = new Byte((byte) CURRENTZOOM);
            }

            return new MapPosition(mapFile.startPosition(), startZoomLevel);
        }
        else
        {
            return getDefaultInitialPosition();
        }
    }

    /**
     * Provides the directory of the map file, by default the Android external storage
     * directory (e.g. sdcard).
     *
     * @return
     */
    protected File getMapFileDirectory()
    {
        return Environment.getExternalStorageDirectory();
    }

    /**
     * Combines map file directory and map file to a map file.
     * This method usually will not need to be changed.
     *
     * @return a map file interface
     */
    protected MapDataStore getMapFile()
    {
        return new MapFile(new File(getMapFileDirectory(), this.getMapFileName()));
    }

    /**
     * The persistable ID is used to store settings information, like the center of the last view
     * and the zoomlevel. By default the simple name of the class is used. The value is not user
     * visibile.
     *
     * @return the id that is used to save this mapview.
     */
    protected String getPersistableId()
    {
        return this.getClass().getSimpleName();
    }

    /**
     * Returns the relative size of a map view in relation to the screen size of the device. This
     * is used for cache size calculations.
     * By default this returns 1.0, for a full size map view.
     *
     * @return the screen ratio of the mapview
     */
    protected float getScreenRatio()
    {
        return 1.0f;
    }

    /**
     * Configuration method to set if a map view activity will have zoom controls.
     *
     * @return true if the map has standard zoom controls.
     */
    protected boolean hasZoomControls()
    {
        return true;
    }

    /**
     * Configuration method to set if map view activity's zoom controls hide automatically.
     *
     * @return true if zoom controls hide automatically.
     */
    protected boolean isZoomControlsAutoHide()
    {
        return true;
    }

    /**
     * initializes the map view position.
     *
     * @param mvp the map view position to be set
     * @return the mapviewposition set
     */
    protected MapViewPosition initializePosition(MapViewPosition mvp)
    {
        final LatLong center = mvp.getCenter();

        if (center.equals(DEFAULT_LOCATION))
        {
            mvp.setMapPosition(this.getInitialPosition());
        }

        mvp.setZoomLevelMax(getZoomLevelMax());
        mvp.setZoomLevelMin(getZoomLevelMin());

        return mvp;
    }

    /**
     * Hook to check for Android Runtime Permissions.
     */
    protected void checkPermissionsAndCreateLayersAndControls()
    {
        createLayers();
        createControls();
    }

    /**
     * Android Fragment life cycle method.
     *
     * @param inflater
     * @param container
     * @param savedInstanceState
     * @return
     */
    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState)
    {
        final View v = inflater.inflate(getLayoutId(), container,
                false);

        /**
         * App Initialization !!
         * behind the scenes, this initialization process gathers a bit of
         * information on your device, such as the screen resolution, that allows mapsforge to
         * automatically adapt the rendering for the device.
         */
        AndroidGraphicFactory.createInstance(this.getActivity().getApplication());

        createSharedPreferences();
        createMapViews(v);
        createTileCaches();
        checkPermissionsAndCreateLayersAndControls();

        return v;
    }

    /**
     * Android Activity life cycle method.
     */
    @Override
    public void onPause()
    {
        mapView.getModel().save(this.preferencesFacade);
        this.preferencesFacade.save();
        super.onPause();
    }

    /**
     * Android Activity life cycle method.
     */
    @Override
    public void onDestroy()
    {
        mapView.destroyAll();
        AndroidGraphicFactory.clearResourceMemoryCache();
        tileCaches.clear();
        super.onDestroy();
    }

    /**
     * Hook to purge tile caches.
     * By default we purge every tile cache that has been added to the tileCaches list.
     */
    protected void purgeTileCaches()
    {
        for (TileCache tileCache : tileCaches)
        {
            tileCache.purge();
        }

        tileCaches.clear();
    }

    protected void redrawLayers()
    {
        mapView.getLayerManager().redrawLayers();
    }

    /**
     * Creates a map view using an XML layout file supplied by getLayoutId() and finds
     * the map view component inside it with getMapViewId().
     *
     * @return the Android MapView for this activity.
     */
    protected MapView getMapView(View v)
    {
        return (MapView) v.findViewById(getMapViewId());
    }

    /**
     * Override to enable hill shading.
     *
     * @return null or the HillsRenderConfig to use (defining height model path and algorithm)
     */
    protected HillsRenderConfig getHillsRenderConfig()
    {
        return null;
    }
}

Надеюсь, это поможет, даже если уже поздно.

Другие вопросы по тегам