42

Android用のGoogleマップAPIを使用するアプリケーションを構築しています。

MapController と MapView があり、以下を使用して組み込みのズーム コントロールを有効にします。

mapView.setBuiltInZoomControls(true);

ユーザーが実際にマップをズームしたときにイベントを取得したいのですが、どうすればよいですか? ズーム レベルの変化を検出できるようなイベントや一般的なイベントは見つかりません。

アップデート

mapView.getZoomControls() は非推奨です。ドキュメントでは、代わりに mapView.setBuiltInZoomControls(bool) を使用することを提案しています。これは問題ありませんが、組み込みのズーム コントロールからイベントを処理する方法がわかりません。

4

11 に答える 11

40

Google Maps Android API v2 では、次のようにGoogleMap.OnCameraChangeListenerを使用できます。

mMap.setOnCameraChangeListener(new OnCameraChangeListener() {

    private float currentZoom = -1;

    @Override
    public void onCameraChange(CameraPosition pos) {
        if (pos.zoom != currentZoom){
            currentZoom = pos.zoom;
            // do you action here
        }
    }
});
于 2013-02-23T23:04:01.723 に答える
25

Android Handlerを使用して、ズームがいつ変更されたかを検出するために、ズーム値の独自の基本的な「ポーリング」を実装できます。

実行可能なイベントに次のコードを使用します。ここで処理を行う必要があります。

private Handler handler = new Handler();

public static final int zoomCheckingDelay = 500; // in ms

private Runnable zoomChecker = new Runnable()
{
    public void run()
    {
        checkMapIcons();

        handler.removeCallbacks(zoomChecker); // remove the old callback
        handler.postDelayed(zoomChecker, zoomCheckingDelay); // register a new one
    }
};

を使用してコールバック イベントを開始します。

protected void onResume()
{
    super.onResume();
    handler.postDelayed(zoomChecker, zoomCheckingDelay);
}

を使用してアクティビティを終了するときに停止します

protected void onPause()
{
    super.onPause();
    handler.removeCallbacks(zoomChecker); // stop the map from updating
}

より長い記事が必要な場合は、これが基づいている記事をここで見つけることができます

于 2011-03-17T21:05:13.270 に答える
7

onZoom 関数リスナーを持つこのライブラリを使用します。http://code.google.com/p/mapview-overlay-manager/ . うまくいきます。

于 2011-03-16T11:50:51.870 に答える
5

OPが言ったように、onUserInteractionEventを使用して自分でテストしてください。

于 2010-01-07T01:08:56.807 に答える
5

ここにきれいな解決策があります。この TouchOverlay プライベート クラスをアクティビティと onZoom というメソッド (この内部クラスによって呼び出される) に追加するだけです。

このTouchOverlayをmapViewに追加する必要があることに注意してください。

mapView.getOverlays().add(new TouchOverlay());

これは、ユーザーがマップにタッチするたびにズーム レベルを追跡し、ズーム レベルが変更された場合は (ズーム レベルを使用して) onZoom メソッドを起動します。

   private class TouchOverlay extends com.google.android.maps.Overlay {
            int lastZoomLevel = -1;

            @Override
            public boolean onTouchEvent(MotionEvent event, MapView mapview) {
                if (event.getAction() == 1) {
                    if (lastZoomLevel == -1)
                        lastZoomLevel = mapView.getZoomLevel();

                    if (mapView.getZoomLevel() != lastZoomLevel) {
                        onZoom(mapView.getZoomLevel());
                        lastZoomLevel = mapView.getZoomLevel();
                    }
                }
                return false;
            }
        }

        public void onZoom(int level) {
            reloadMapData(); //act on zoom level change event here
        }
于 2012-05-14T13:55:41.257 に答える
4

使用できます

ZoomButtonsController zoomButton = mapView.getZoomButtonsController();
zoomButton.setOnZoomListener(listener);

それが役に立てば幸い

于 2010-06-23T01:04:37.307 に答える
4

Android APIでは、setOnZoomInClickListener() を持つZoomControlsを使用することをお勧めします。

これらのズーム コントロールを追加するには、次のようにします。

ZoomControls mZoom = (ZoomControls) mapView.getZoomControls();

次に、mZoom をレイアウトに追加します。

于 2010-01-06T14:36:53.767 に答える
3

I think there might another answer to this question. The Overlay class draw method is called after any zoom and I believe this is called after the zoom level changes. Could you not architect your app to take advantage of this? You could even use a dummy overlay just to detect this if you wanted to. Would this not be more efficient than using a runnable with a delay?

@Override
public boolean draw (Canvas canvas, MapView mapView, boolean shadow, long when) {
    int zoomLevel = mapView.getZoomLevel();
    // do what you want with the zoom level
    return super.draw(canvas, mapView, shadow, when);
}
于 2012-03-17T16:19:49.650 に答える
2

V2.0 より前のバージョンでは、MapView を拡張するクラスを作成しました。このクラスは、マップ領域が変化し始めたとき (onRegionBeginChange) と変化しなくなったとき (onRegionEndChange) にイベントでリスナーに警告します。

import android.content.Context;
import android.util.AttributeSet;
import android.view.MotionEvent;

import com.google.android.maps.GeoPoint;
import com.google.android.maps.MapView;

public class ExMapView extends MapView {
    private static final String TAG = ExMapView.class.getSimpleName();
    private static final int DURATION_DEFAULT = 700;

    private OnRegionChangedListener onRegionChangedListener;
    private GeoPoint previousMapCenter;
    private int previousZoomLevel;
    private int changeDuration; // This is the duration between when the user stops moving the map around and when the onRegionEndChange event fires.
    private boolean isTouched = false;
    private boolean regionChanging = false;

    private Runnable onRegionEndChangeTask = new Runnable() {
        public void run() {
            regionChanging = false;
            previousMapCenter = getMapCenter();
            previousZoomLevel = getZoomLevel();
            if (onRegionChangedListener != null) {
                onRegionChangedListener.onRegionEndChange(ExMapView.this, previousMapCenter, previousZoomLevel);
            }
        }
    };

    public ExMapView(Context context, AttributeSet attrs) {
        super(context, attrs);
        init();
    }

    public ExMapView(Context context, AttributeSet attrs, int defStyle) {
        super(context, attrs, defStyle);
        init();
    }

    public ExMapView(Context context, String apiKey) {
        super(context, apiKey);
        init();
    }

    @Override
    public boolean onTouchEvent(MotionEvent event) {
        isTouched = event.getAction() != MotionEvent.ACTION_UP;
        return super.onTouchEvent(event);
    }

    @Override
    public void computeScroll() {
        super.computeScroll();

        // If the map region is still changing (user is still scrolling or zooming), reset timer for onRegionEndChange.
        if ((!isTouched && !getMapCenter().equals(previousMapCenter)) || (previousZoomLevel != getZoomLevel())) {

            // If the region has just begun changing, fire off onRegionBeginChange event.
            if (!regionChanging) {
                regionChanging = true;
                if (onRegionChangedListener != null) {
                    onRegionChangedListener.onRegionBeginChange(this, previousMapCenter, previousZoomLevel);
                }
            }

            // Reset timer for onRegionEndChange.
            removeCallbacks(onRegionEndChangeTask);
            postDelayed(onRegionEndChangeTask, changeDuration);
        }
    }

    private void init() {
        changeDuration = DURATION_DEFAULT;
        previousMapCenter = getMapCenter();
        previousZoomLevel = getZoomLevel();
    }

    public void setOnRegionChangedListener(OnRegionChangedListener listener) {
        onRegionChangedListener = listener;
    }

    public void setChangeDuration(int duration) {
        changeDuration = duration;
    }

    public interface OnRegionChangedListener {
        public abstract void onRegionBeginChange(ExMapView exMapView, GeoPoint geoPoint, int zoomLevel);
        public abstract void onRegionEndChange(ExMapView exMapView, GeoPoint geoPoint, int zoomLevel);
    }
}
于 2014-02-07T04:15:25.147 に答える
2

U は、このような Zoom CControl を持つことができます。

ここで、ズームコントロールを実装する方法のコードを添付しています。

1>> XML ファイルは次のようになります。

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" 
    android:layout_width="fill_parent" 
    android:layout_height="fill_parent">

    <com.google.android.maps.MapView 
        android:id="@+id/mapView"
        android:layout_width="fill_parent"
        android:layout_height="fill_parent"
        android:enabled="true"
        android:clickable="true"
        android:apiKey="0l4sCTTyRmXTNo7k8DREHvEaLar2UmHGwnhZVHQ"
        />

    <LinearLayout android:id="@+id/zoom" 
        android:layout_width="wrap_content" 
        android:layout_height="wrap_content" 
        android:layout_alignParentBottom="true" 
        android:layout_centerHorizontal="true" 
        /> 

</RelativeLayout>

2>> oncreate メソッド内の主なアクティビティで、次のことを行います。ここで、mapView でビューを膨らませています

mapView = (MapView) findViewById(R.id.mapView);
        LinearLayout zoomLayout = (LinearLayout)findViewById(R.id.zoom);  
        View zoomView = mapView.getZoomControls(); 

        zoomLayout.addView(zoomView, 
            new LinearLayout.LayoutParams(
                LayoutParams.WRAP_CONTENT, 
                LayoutParams.WRAP_CONTENT)); 
        mapView.displayZoomControls(true);

................................................................... .........これは、APIの組み込みズームコントロールを使用する方法です..

このようにズームインとズームアウトを実装するための独自のカスタムコードを作成できます...

public boolean onKeyDown(int keyCode, KeyEvent event) 
{
    MapController mc = mapView.getController(); 
    switch (keyCode) 
    {
        case KeyEvent.KEYCODE_3:
            mc.zoomIn();
            break;
        case KeyEvent.KEYCODE_1:
            mc.zoomOut();
            break;
    }
    return super.onKeyDown(keyCode, event);
}    

..これが私が実装した方法です...

ありがとう..Rakesh

于 2010-08-27T10:55:29.360 に答える
1

上記のものを混ぜて使いました。タイマーを使用して 0.5 秒ごとにスレッドを開始すると、マップが非常にぎくしゃくすることがわかりました。おそらく、毎回いくつかの If ステートメントを使用していたためです。そのため、onUserInteraction から 500 ミリ秒のポスト遅延でスレッドを開始しました。これにより、スレッドが実行を開始する前に zoomLevel を更新するのに十分な時間が与えられるため、500 ミリ秒ごとにスレッドを実行しなくても正しいズームレベルを取得できます。

public void onUserInteraction() {
    handler.postDelayed(zoomChecker, zoomCheckingDelay);
}
于 2011-04-29T12:48:05.083 に答える