我想使Google Map v2信息窗口成为生活视图,因此我将SupportMapFragment包装在自定义RelativeLayout中,如下所示:

R.layout.info_window:

<?xml version="1.0" encoding="utf-8"?>
<com.Amazing.GoogleMapV2.MapWrapperLayout
        xmlns:android="http://schemas.android.com/apk/res/android"
        android:layout_width="fill_parent"
        android:layout_height="fill_parent"
        android:id="@+id/map_relative_layout">
    <fragment
            android:id="@+id/map"
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            class="com.google.android.gms.maps.SupportMapFragment" />
</com.Amazing.GoogleMapV2.MapWrapperLayout>


我重写了dispatchTouchEvent,将motionEvent位置偏移到infowindow视图坐标,然后将dispatchTouchEvent偏移到infoWindow:
MapWrapperLayout:

public class MapWrapperLayout extends RelativeLayout {
    /**
     * Reference to a GoogleMap object
     */
    private GoogleMap map;

    /**
     * Vertical offset in pixels between the bottom edge of our InfoWindow
     * and the marker position (by default it's bottom edge too).
     * It's a good idea to use custom markers and also the InfoWindow frame,
     * because we probably can't rely on the sizes of the default marker and frame.
     */
    private int bottomOffsetPixels;

    /**
     * A currently selected marker
     */
    private Marker marker;

    /**
     * Our custom view which is returned from either the InfoWindowAdapter.getInfoContents
     * or InfoWindowAdapter.getInfoWindow
     */
    private View infoWindow;

    public MapWrapperLayout(Context context) {
        super(context);
    }

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

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

    /**
     * Must be called before we can route the touch events
     */
    public void init(GoogleMap map, int bottomOffsetPixels) {
        this.map = map;
        this.bottomOffsetPixels = bottomOffsetPixels;
    }

    /**
     * Best to be called from either the InfoWindowAdapter.getInfoContents
     * or InfoWindowAdapter.getInfoWindow.
     */
    public void setMarkerWithInfoWindow(Marker marker, View infoWindow) {
        this.marker = marker;
        this.infoWindow = infoWindow;
    }

    @Override
    public boolean dispatchTouchEvent(MotionEvent ev) {
        // Make sure that the infoWindow is shown and we have all the needed references
        if (marker != null && marker.isInfoWindowShown() && map != null && infoWindow != null) {
            // Get a marker position on the screen
            Point point = map.getProjection().toScreenLocation(marker.getPosition());

            // Make a copy of the MotionEvent and adjust it's location
            // so it is relative to the infoWindow left top corner
            MotionEvent copyEv = MotionEvent.obtain(ev);
            copyEv.offsetLocation(
                    -point.x + (infoWindow.getWidth() / 2),
                    -point.y + infoWindow.getHeight() + bottomOffsetPixels);
            Log.e("~~~~~~~~~~dispatchTouchEvent  copyEv", copyEv.toString());

            //Dispatch the adjusted MotionEvent to the infoWindow
            if (infoWindow.dispatchTouchEvent(copyEv)) {
                Log.e("infoWindow.dispatchTouchEvent(copyEv)", "True");
                Log.e("~~~~~~~~~~dispatchTouchEvent  point", point.toString());
                Log.e("~~~~~~~~~~dispatchTouchEvent  ev", ev.toString());
                Log.e("~~~~~~~~~~dispatchTouchEvent  copyEv", copyEv.toString());
                return true;
            }
        }
        // If the infoWindow consumed the touch event, then just return true.
        // Otherwise pass this event to the super class and return it's result
        return super.dispatchTouchEvent(ev);
    }
}


R.layout.info_window:

<?xml version="1.0" encoding="utf-8"?>

<LinearLayout
        xmlns:android="http://schemas.android.com/apk/res/android"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:gravity="center_vertical">

    <LinearLayout
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:orientation="vertical"
            android:layout_marginRight="10dp">

        <TextView
                android:id="@+id/title"
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:textSize="18sp"
                android:text="Title"/>

        <TextView
                android:id="@+id/snippet"
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:text="snippet"/>

    </LinearLayout>

    <Button
            android:id="@+id/button"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="Button" android:clickable="true" android:enabled="true" android:focusable="true"
            android:focusableInTouchMode="true"/>

</LinearLayout>


然后MainActivity将R.layout.info_window和setOnClickListener膨胀到info_window上的infoButton:
主要活动:

    public class MainActivity extends SherlockFragmentActivity {
        private ViewGroup infoWindow;
        private TextView infoTitle;
        private TextView infoSnippet;
        private Button infoButton;
    //    private OnInfoWindowElemTouchListener infoButtonListener;

        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.main);

            final SupportMapFragment mapFragment =
                    (SupportMapFragment)getSupportFragmentManager().findFragmentById(R.id.map);
            final MapWrapperLayout mapWrapperLayout =
                    (MapWrapperLayout)findViewById(R.id.map_relative_layout);
            final GoogleMap map = mapFragment.getMap();

            // MapWrapperLayout initialization
            // 39 - default marker height
            // 20 - offset between the default InfoWindow bottom edge and it's content bottom edge
            mapWrapperLayout.init(map, getPixelsFromDp(this, 39 + 20));

            // We want to reuse the info window for all the markers,
            // so let's create only one class member instance
            this.infoWindow = (ViewGroup)getLayoutInflater().inflate(R.layout.info_window, null);
            this.infoTitle = (TextView)infoWindow.findViewById(R.id.title);
            this.infoSnippet = (TextView)infoWindow.findViewById(R.id.snippet);
            this.infoButton = (Button)infoWindow.findViewById(R.id.button);
            // Setting custom OnTouchListener which deals with the pressed state
            // so it shows up
    //        this.infoButtonListener = new OnInfoWindowElemTouchListener(infoButton,
    //                R.drawable.btn_normal, R.drawable.btn_pressed)
    //        {
    //            @Override
    //            protected void onClickConfirmed(View v, Marker marker) {
    //                // Here we can perform some action triggered after clicking the button
    //                Toast.makeText(MainActivity.this, marker.getTitle() + "'s button clicked!", Toast.LENGTH_SHORT).show();
    //            }
    //        };
            this.infoButton.setOnClickListener(new View.OnClickListener() {
                @Override
                public void onClick(View view) {
                    Toast.makeText(MainActivity.this, "~~~~~ Button clicked!", Toast.LENGTH_SHORT).show();
                }
            });

            map.setInfoWindowAdapter(new GoogleMap.InfoWindowAdapter() {
                @Override
                public View getInfoWindow(Marker marker) {
                    return null;
                }

                @Override
                public View getInfoContents(Marker marker) {
                    // Setting up the infoWindow with current's marker info
                    infoTitle.setText(marker.getTitle());
                    infoSnippet.setText(marker.getSnippet());
    //                infoButtonListener.setMarker(marker);

                    // We must call this to set the current marker and infoWindow references
                    // to the MapWrapperLayout
                    mapWrapperLayout.setMarkerWithInfoWindow(marker, infoWindow);
                    return infoWindow;
                }
            });

            // Let's add a couple of markers
            map.animateCamera(CameraUpdateFactory.newLatLng(new LatLng(50.08, 14.43)));
            map.addMarker(new MarkerOptions()
                    .title("Prague")


   .snippet("Czech Republic")
                .position(new LatLng(50.08, 14.43)));

        map.addMarker(new MarkerOptions()
                .title("Paris")
                .snippet("France")
                .position(new LatLng(48.86,2.33)));

        Marker m = map.addMarker(new MarkerOptions()
                .title("London")
                .snippet("United Kingdom")
                .position(new LatLng(51.51,-0.1)));
        m.showInfoWindow();
    }

    public static int getPixelsFromDp(Context context, float dp) {
        final float scale = context.getResources().getDisplayMetrics().density;
        return (int)(dp * scale + 0.5f);
    }
}


这些代码的行为很奇怪,当活动onResume()不是立即调用时,按钮的onClick将被调用,
但是onTouch事件会立即被调用。
为什么?

最佳答案

我认为这可能与其他问题有关。

Runnable is posted successfully but not run

问题是在View.onTouchEvent(MotionEvent event)中,当ACTION_UP事件发生时,而不是仅调用performClick()时,Android会以PerformClick可运行的方式调用post()。由于您的视图未附加到窗口,因此该可运行项无法运行。

要解决此问题,只需将视图添加到MapWrapperLayout。您甚至可以将其放在屏幕之外,以免影响地图的外观。

public void setMarkerWithInfoWindow ( Marker marker, View infoWindow )
{
    this.marker = marker;

    if ( this.infoWindow != null )
    {
        this.removeView ( this.infoWindow );
    }
    this.infoWindow = infoWindow;

    // Force the view to render WAY off screen, so we don't actually see it.
    RelativeLayout.LayoutParams params = new RelativeLayout.LayoutParams (
            LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT );
    params.leftMargin = 100000;

    this.addView ( infoWindow, params );
}

09-11 22:40