谷歌地图Android API 2 - 用于MapBox实现的瓦片混合TileProvider / URLTileProvider

5
我想实现一个具备以下功能的TileProvider:
  • Map 请求 Tile
  • TileProvider 检查所需的 Tile 是否可以离线使用
  • 如果可以,TileProvider 返回该 Tile
  • 无论是否可用,TileProvider 都会在线上检查新 Tile 并将其下载到外部存储器中的 Cache
  • TileProvider 提供新的 Tile 给地图,并更新它。
这是我当前的 UrlTileProvider 实现:
public class MapBoxOnlineTileProvider extends UrlTileProvider {

    private static final String[] FORMATS;

    static {
        final String[] servers = new String[] { "a", "b", "c", "d" };
        final String[] formats = new String[servers.length];
        for (int i = 0; i < servers.length; i++) {
            formats[i] = String.format(
                    "%%s://%s.tiles.mapbox.com/v3/%%s/%%d/%%d/%%d.png",
                    servers[i]);
        }
        FORMATS = formats;
    }

    // ------------------------------------------------------------------------
    // Instance Variables
    // ------------------------------------------------------------------------

    private String mMapIdentifier;

    // ------------------------------------------------------------------------
    // Constructors
    // ------------------------------------------------------------------------



    public MapBoxOnlineTileProvider(final String mapIdentifier) {
        super(256, 256);
        this.mMapIdentifier = mapIdentifier;
    }

    // ------------------------------------------------------------------------
    // Public Methods
    // ------------------------------------------------------------------------

    public String getMapIdentifier() {
        return this.mMapIdentifier;
    }

    public void setMapIdentifier(final String anIdentifier) {
        this.mMapIdentifier = anIdentifier;
    }

    @Override
    public URL getTileUrl(final int x, final int y, final int z) {
        final String f = FORMATS[new Random().nextInt(FORMATS.length)];
        final String p = "http";
        try {
            return new URL(String.format(f, p, this.mMapIdentifier, z, x, y));
        } catch (final MalformedURLException e) {
            e.printStackTrace();
            return null;
        }
    }
}

然而,这只会在内存中缓存它,所以如果用户关闭我的应用程序,它就不会被缓存了,在下载 Tiles 之前,将显示默认的谷歌地图。 UrlTileProvidergetTile 实现为 final,因此我无法重写它。 TileProvider 没有 getTileURL 方法,而且我无法获得 UrlTileProvider 的源代码来重新编写解决方案。
我该如何解决这个问题?有没有办法做到我想做的事情?为什么 getTile 是 final 的?我有什么遗漏的吗?
1个回答

1
使用第二类包装了第一类,问题已解决。
尽管我不喜欢包装器,但这是我能想到的最好方法,我必须说它运行得非常好。
public class MapBoxMixedTileProvider implements TileProvider {

    private final MapBoxOnlineTileProvider provider;
    private final Context context;

    public MapBoxMixedTileProvider(final Context context, final String mapId) {
        provider = new MapBoxOnlineTileProvider(mapId);
        this.context = context;
    }

    @Override
    public Tile getTile(final int arg0, final int arg1, final int arg2) {
        final URL url = provider.getTileUrl(arg0, arg1, arg2);
        // Remove the "http://a." from the url, then replace all the '/' and '.'
        // with
        // '_', then replace the final "_png" with ".png", so openFileInput
        // doesn't cry a lot.
        final String finalUrl = (url.toString().substring(9)).replace('/', '_')
                .replace('.', '_').replace("_png", ".png");
        System.out.println(finalUrl);
        boolean contained = false;
        for (final String file : context.fileList()) {
            if (file.contains(finalUrl)) {
                contained = true;
                break;
            }
        }
        try {
            if (contained) {
                final FileInputStream fis = context.openFileInput(finalUrl);
                return new Tile(256, 256, byteArrayFromInputStream(fis));
            } else {
                final FileOutputStream fos = context.openFileOutput(finalUrl,
                        Context.MODE_PRIVATE);
                final Tile t = provider.getTile(arg0, arg1, arg2);
                fos.write(t.data);
                fos.flush();
                fos.close();
                return t;
            }
        } catch (final IOException ex) {
            ex.printStackTrace();
            return null;
        }
    }

    private byte[] byteArrayFromInputStream(final FileInputStream is)
            throws IOException {
        final BufferedInputStream bis = new BufferedInputStream(is);
        final ArrayList<Integer> bytes = new ArrayList<Integer>();
        int current = 0;
        while ((current = bis.read()) != -1) {
            bytes.add(Integer.valueOf(current));
        }
        final byte[] bs = new byte[bytes.size()];
        for (int i = 0; i < bytes.size(); i++) {
            bs[i] = bytes.get(i).byteValue();
        }
        bis.close();
        is.close();
        return bs;
    }

}

这样,为了使 TileProvider 起作用,我只需要这样做:

final MapBoxMixedTileProvider provider = new MapBoxMixedTileProvider(
                        this, MAPBOX_MAP_IDENTIFIER);
//zIndex is set to -4 so Polyines and Polygons get shown OVER and not under the Overlay
final TileOverlayOptions overlay = new TileOverlayOptions().tileProvider(provider).zIndex(-4);
tileOverlay = mMap.addTileOverlay(overlay);

我知道我的byteArrayFromInputStream方法非常低效。欢迎对我的代码进行任何升级。


什么是MapBoxOnlineTileProvider? - pablobaldez

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接