0

カスタムオーバーレイと多数のマーカーを含むマップがあります。ズームの変更が発生したときにマーカーのグループ化を行っているため、オーバーレイをクリアし、新しいズーム用にアイテムを再グループ化してから、もう一度マップ上に配置する必要があります。私は物事をスムーズにするためにバックラウンドスレッドでこれを行っています。クリックしてズームすると、マーカーが消え、ビジーインジケーターが隅に表示され、新しいマーカーが地図に表示されます。すべてがスムーズですが、2回続けてズームインするか、ズームインしてすぐにズームアウトすると、同時変更の例外が発生します。私はこれについてたくさん読みましたが、まだ解決策はありません。たとえば、マップのモイフィケーションをUI上にあるonPostExecuteに移動すると、これは防止されますが、マーカーが描画されるまでマップがフリーズし、滑らかさが失われます。アイテム化されたオーバーレイ同期リストにマーカーリストを作成しようとしましたが、役に立ちませんでした。どうすればこれを防ぐことができますか?

これは私がズーム変更と呼んでいるものです:

private void drawItems() {
  final MyMapView mw = this.mapView;

  //remove existing first
  //mw.getOverlays().remove(stuffOverlay);
  stuffOverlay.clearAll();
  //mw.invalidate();

  new AsyncTask<String, Integer, String>() {
    @Override
    protected void onPreExecute() {
      super.onPreExecute();
      ((ProgressBar)findViewById(R.id.mapProgressBar)).setVisibility(View.VISIBLE);
    }

    @Override
    protected String doInBackground(String... params) {
      //get items on map
      HashMap<String, JSONArray> groupedItems = getItemsForZoom(mw.getZoomLevel());

      if (groupedItems==null)
        return "done";

      //create a copy of it sometimes it was throwing concurentmodificationexception
      HashMap<String, JSONArray> groupedItemsCopy = new HashMap<String, JSONArray> (groupedItems);

      Iterator it = groupedItemsCopy.entrySet().iterator();
      while (it.hasNext()) { //loop over markers
        try {
          //get item 
          HashMap.Entry<String, Object> pair = (HashMap.Entry)it.next();
          JSONArray itemsInMarker = (JSONArray) pair.getValue();
          JSONObject itemData = (JSONObject) itemsInMarker.get(0);

          //get truncated coordinates
          String coordsKey = pair.getKey();
          String[] coordsSplitted = coordsKey.split("_");

          //add first marker to map
          GeoPoint gp = new GeoPoint((int)(Double.parseDouble(coordsSplitted[0])*1E6), (int)(Double.parseDouble(coordsSplitted[1])*1E6));
          OverlayItem markerOverlay = new OverlayItem(gp, "marker", "snipper text");
          Drawable markerPic = ServerCall.drawableFromUrl(((JSONObject)((JSONObject)itemData.get("picture")).get("1")).get("picture_full")+"_0.png");
          //markerPic.setBounds(0, 0, markerPic.getIntrinsicWidth(), markerPic.getIntrinsicHeight());
          markerPic.setBounds(-25, -25, 50, 50);
          markerOverlay.setMarker(markerPic);
          stuffOverlay.addOverlay(markerOverlay, itemData);

          if (stuffOverlay.size() > 0) {
            mapView.getOverlays().add(stuffOverlay);
          }
        } catch (JSONException e) {
          e.printStackTrace();
        } catch (IOException e) {
          e.printStackTrace();
        }
      }
      return "done";
    }

    @Override
    protected void onPostExecute(String result) {
      super.onPostExecute(result);
      ((ProgressBar)findViewById(R.id.mapProgressBar)).setVisibility(View.GONE);
      mapView.invalidate();
    }
  }.execute();
}

これは私の項目別オーバーレイです:

private class StuffOverlay extends ItemizedOverlay {
  private List<OverlayItem> mOverlays = Collections.synchronizedList(new ArrayList<OverlayItem>());
  private ArrayList<JSONObject> mStuff = new ArrayList<JSONObject>();
  public int currentItem;
  Map mMap;

  public StuffOverlay(Drawable defaultMarker, Map map) {
    super(boundCenterBottom(defaultMarker));
    mMap = map;
    populate(); //keep here
  }

  public void addOverlay(OverlayItem overlay, JSONObject stuffData) {
    synchronized (mOverlays) {
      mOverlays.add(overlay);
    }

    mStuff.add(stuffData);
    setLastFocusedIndex(-1);  //keep here
    populate();
  }

  public void clearAll() {
    synchronized (mOverlays) {
      mOverlays.clear();
    }

    setLastFocusedIndex(-1); //keep here
    populate();
  }

  @Override
  protected OverlayItem createItem(int i) {
    synchronized (mOverlays) {
      return mOverlays.get(i);
    }
  }

  @Override
  public int size() {
    synchronized (mOverlays) {
      return mOverlays.size();
    }
  }

  public void draw(Canvas canvas, MapView mapView, boolean shadow) {
    if (!shadow) {
      super.draw(canvas, mapView, false);
    }
  }
}

そして、これは私が得ているものです; o(

05-30 20:45:35.485: W/dalvikvm(1139): threadid=3: thread exiting with uncaught exception (group=0x4001b188)
05-30 20:45:35.485: E/AndroidRuntime(1139): Uncaught handler: thread main exiting due to uncaught exception
05-30 20:45:35.495: E/AndroidRuntime(1139): java.util.ConcurrentModificationException
05-30 20:45:35.495: E/AndroidRuntime(1139):     at java.util.AbstractList$SimpleListIterator.next(AbstractList.java:64)
05-30 20:45:35.495: E/AndroidRuntime(1139):     at com.google.android.maps.OverlayBundle.draw(OverlayBundle.java:44)
05-30 20:45:35.495: E/AndroidRuntime(1139):     at com.google.android.maps.MapView.onDraw(MapView.java:476)
05-30 20:45:35.495: E/AndroidRuntime(1139):     at android.view.View.draw(View.java:6535)
05-30 20:45:35.495: E/AndroidRuntime(1139):     at android.view.ViewGroup.drawChild(ViewGroup.java:1531)
05-30 20:45:35.495: E/AndroidRuntime(1139):     at android.view.ViewGroup.dispatchDraw(ViewGroup.java:1258)
4

1 に答える 1

1

問題は、ここにあるこれらの行に関係している可能性があると思います。

      if (stuffOverlay.size() > 0) {
        mapView.getOverlays().add(stuffOverlay);
      }

オブジェクトをマップオーバーレイ配列に追加する必要があるのはstuffOverlay1回だけで、stuffOverlayにアイテムが含まれる前に追加することもできます。mapView.getOverlays().add(stuffOverlay);stuffOverlayがインスタンス化されるポイントに移動します。

于 2012-05-31T03:15:11.163 に答える