2

アプリケーションのカスタム MapView を使用しています。多くの GPS データポイントが保存されたデータベースがあり、現在、オーバーレイを使用してそれらをロードし、オーバーレイを使用して MapView で .addAll を実行しています。

ただし、数百のデータ ポイントを MapView のパスとしてロードするのに最大 2 分かかります。

データ ポイントをロードするコードは次のとおりです。

    private void loadMap() {        
    //Load the specific data from the database
    DBAdapter db = new DBAdapter(this);
    try {
        db.open();
        Cursor mDataPoints = db.db.rawQuery("SELECT * FROM GPS_DATA)", null);
        if ((mDataPoints != null) && (!mDataPoints.isClosed()) && (mDataPoints.moveToFirst())) {                
            mapMain.setFinishedLoading(false);
            mapMain.getOverlays().clear();

            if ((!mDataPoints.isClosed()) && (mDataPoints.moveToFirst())) {
                while (!mDataPoints.isAfterLast()) {

                    this.drawPath(db, 
                                  DBAdapter.formatLong(mDataPoints, "data_id"), 
                                  DBAdapter.formatString(mDataPoints, "data_name"),
                                  DBAdapter.formatString(mDataPoints, "data_type"));                        

                    mDataPoints.moveToNext();
                }
            }

            mapMain.refreshDrawableState();
            mapMain.setEnabled(true);
        }
    } finally {
        db.close();
    }
}

public void drawPath(DBAdapter db, long pDataID, String pDataName, String pDataType) {  

    Cursor mData = db.getPointDataByDataID(pDataID);
    if ((!mData.isClosed()) && (mData.moveToFirst())) {
        boolean vFirst = true;
        int i = 0;
        GeoPoint startGP;
        GeoPoint geoPoint1;
        GeoPoint geoPoint2 = null;

        while (!mData.isAfterLast()) {
            if (vFirst) {
                startGP = new GeoPoint((int) (DBAdapter.formatDouble(mData, "latitude") * 1E6), (int) (DBAdapter.formatDouble(mData, "longitude") * 1E6));
                mapMain.getOverlays().add(new Map_Route_Overlay(startGP, startGP, this.returnPathColor(pDataType), pDataName));
                vFirst = false;
                geoPoint2 = startGP;
            } else {
                geoPoint1 = geoPoint2;

                geoPoint2 = new GeoPoint((int) (DBAdapter.formatDouble(mData, "latitude") * 1E6), (int) (DBAdapter.formatDouble(mData, "longitude") * 1E6));
                if (geoPoint2.getLatitudeE6() != 22200000) {
                    mapMain.getOverlays().add(new Map_Route_Overlay(geoPoint1, geoPoint2, this.returnPathColor(pDataType), pDataName));
                }
            }

            i += 1;
            mData.moveToNext();
        }

        if (geoPoint2 != null) { 
            mapMain.getOverlays().add(new Map_Route_Overlay(geoPoint2, geoPoint2, this.returnDataColor(pDataType), pDataName));
        }
    }
    mData.close();
}
public int returnDataColor(String pType) {
    int vReturnData = Color.BLUE;

    if (pType.trim().equals("S")) {
        vReturnData = Color.RED;
    }

    return vReturnData;
}

Map_Route_Overlay クラスのコードは次のとおりです。

public final class Map_Route_Overlay extends Overlay {
  private static String TAG = CoreFunctions.APP_TAG + "_MapRouteOverlay";

  private GeoPoint geoPoint1;
  private GeoPoint geoPoint2;
  private int defaultColor = 999;
  private String dataName = "";

  public Map_Route_Overlay(GeoPoint pGeoPoint1, GeoPoint pGeoPoint2, int pColor, String pText) {
    this.geoPoint1 = pGeoPoint1;
    this.geoPoint2 = pGeoPoint2;
    this.defaultColor = pColor;
    this.dataName = pText;
  }

  public void setDataName(String pName) {
    this.dataName = pName;
  }

  @Override
  public boolean draw (Canvas canvas, MMMapView mapView, boolean shadow, long when) {
    Projection projection = mapView.getProjection();
    if (shadow == false) {
        Paint paint = new Paint();
        paint.setAntiAlias(true);
        Point point = new Point();
        projection.toPixels(this.geoPoint1, point);
        paint.setColor(this.defaultColor);
        Point point2 = new Point();
        projection.toPixels(this.geoPoint2, point2);
        paint.setStrokeWidth(2);
        paint.setAlpha(this.defaultColor == Color.parseColor("#6C8715")?220:120);
        canvas.drawLine(point.x, point.y, point2.x,point2.y, paint);
    }

    return super.draw(canvas, mapView, shadow, when);
  }

}

スレッドを使用してデータをロードするなど、いくつかのオプションを試しましたが、完了時にオーバーレイをロードするように MapView に通知するのは悪夢になりました。

参考までに、MMMapView は私のカスタム MapView コンポーネントであり、現時点では MapView を継承し、2 つの関数を実装しています。

4

2 に答える 2

2

一度に何百ものオーバーレイを画面に表示しようとしてはいけません。パフォーマンスが低下するからです。10 ~ 20 に制限してみてください。それ以上だと、とにかくユーザーにとって忙しすぎます。

また、スレッドを使用してデータベースからアイテムをロードする必要があります。私がしていることは、カスタム ItemizedOverlay を使用し、カスタム OverlayItems をスレッドに追加することです。次に、読み込みが完了したらmapView.postInvalidate();、それらを呼び出して画面に表示します。

于 2011-11-16T17:56:38.630 に答える
1

私はこれを使用しています: http://code.google.com/p/mapview-overlay-manager/

マップ上に表示されているマーカーのみをロードする LazyLoading を提供します。

于 2011-11-16T18:02:15.570 に答える