我在ScrollView
中有一个新的MapFragment
。实际上它是SupportMapFragment
,但无论如何。它有效,但有两个问题:
滚动时,会留下黑色面具。黑色完全覆盖了地图所在的区域,除了+/-缩放按钮所在的孔。见下面的截图。这是在Android 4.0上。
当用户与地图互动以阻止requestDisallowInterceptTouchEvent()
拦截触摸时,视图不会使用ScrollView
,因此如果您尝试在地图中垂直平移,则只会滚动包含ScrollView
。理论上我可以派生一个视图类MapView
并添加该功能,但是如何让MapFragment
使用我自定义的MapView
而不是标准的?{/ p>
答案 0 :(得分:52)
在mapview片段上应用透明图像似乎可以解决问题。这不是最漂亮的,但似乎有效。这是一个显示以下内容的XML片段:
<RelativeLayout
android:id="@+id/relativeLayout1"
android:layout_width="match_parent"
android:layout_height="300dp" >
<fragment
android:id="@+id/map"
android:name="com.google.android.gms.maps.MapFragment"
android:layout_width="fill_parent"
android:layout_height="fill_parent"/>
<ImageView
android:id="@+id/imageView123"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:src="@drawable/temp_transparent" />
</RelativeLayout>
答案 1 :(得分:35)
对我来说,添加透明的ImageView并没有帮助完全删除黑色面具。滚动时,地图的顶部和底部部分仍显示黑色遮罩。
所以它的解决方案,我在this answer中发现了一个小小的变化。 我补充说,
android:layout_marginTop="-100dp"
android:layout_marginBottom="-100dp"
到我的地图片段,因为它是垂直scrollview。所以我的布局现在看起来像这样:
<RelativeLayout
android:id="@+id/map_layout"
android:layout_width="match_parent"
android:layout_height="300dp">
<fragment
android:id="@+id/mapview"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:layout_marginTop="-100dp"
android:layout_marginBottom="-100dp"
android:name="com.google.android.gms.maps.MapFragment"/>
<ImageView
android:id="@+id/transparent_image"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:src="@color/transparent" />
</RelativeLayout>
要解决问题的第二部分,我为主ScrollView设置了requestDisallowInterceptTouchEvent(true)
。当用户触摸透明图像并移动时,我在MotionEvent.ACTION_DOWN
和MotionEvent.ACTION_MOVE
的透明图像上禁用了触摸,以便地图片段可以进行触摸事件。
ScrollView mainScrollView = (ScrollView) findViewById(R.id.main_scrollview);
ImageView transparentImageView = (ImageView) findViewById(R.id.transparent_image);
transparentImageView.setOnTouchListener(new View.OnTouchListener() {
@Override
public boolean onTouch(View v, MotionEvent event) {
int action = event.getAction();
switch (action) {
case MotionEvent.ACTION_DOWN:
// Disallow ScrollView to intercept touch events.
mainScrollView.requestDisallowInterceptTouchEvent(true);
// Disable touch on transparent view
return false;
case MotionEvent.ACTION_UP:
// Allow ScrollView to intercept touch events.
mainScrollView.requestDisallowInterceptTouchEvent(false);
return true;
case MotionEvent.ACTION_MOVE:
mainScrollView.requestDisallowInterceptTouchEvent(true);
return false;
default:
return true;
}
}
});
这对我有用。希望它可以帮助你..
答案 2 :(得分:8)
这可能源于this question导致问题的同一个地方。解决方案是使用透明框架,重量比透明图像轻一点。
答案 3 :(得分:3)
经过大量的R&amp; D:
fragment_one.xml应如下所示:
<?xml version="1.0" encoding="utf-8"?>
<ScrollView xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:id="@+id/scrollViewParent"
android:orientation="vertical" >
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="vertical" >
<RelativeLayout
android:layout_width="match_parent"
android:layout_height="400dip" >
<com.google.android.gms.maps.MapView
android:id="@+id/mapView"
android:layout_width="match_parent"
android:layout_height="match_parent" />
<View
android:id="@+id/customView"
android:layout_width="fill_parent"
android:layout_height="fill_parent"
android:background="@android:color/transparent" />
</RelativeLayout>
<!-- Your other elements are here -->
</LinearLayout>
</ScrollView>
您的Java类FragmentOne.java如下所示:
private GoogleMap mMap;
private MapView mapView;
private UiSettings mUiSettings;
private View customView
<强> onCreateView 强>
mapView = (MapView) rootView.findViewById(R.id.mapView);
mapView.onCreate(savedInstanceState);
if (mapView != null) {
mMap = mapView.getMap();
mMap.setMapType(GoogleMap.MAP_TYPE_NORMAL);
mUiSettings = mMap.getUiSettings();
mMap.setMyLocationEnabled(true);
mUiSettings.setCompassEnabled(true);
mUiSettings.setMyLocationButtonEnabled(false);
}
scrollViewParent = (ScrollView)rootView.findViewById(R.id.scrollViewParent);
customView = (View)rootView.findViewById(R.id.customView);
customView.setOnTouchListener(new View.OnTouchListener() {
@Override
public boolean onTouch(View v, MotionEvent event) {
int action = event.getAction();
switch (action) {
case MotionEvent.ACTION_DOWN:
// Disallow ScrollView to intercept touch events.
scrollViewParent.requestDisallowInterceptTouchEvent(true);
// Disable touch on transparent view
return false;
case MotionEvent.ACTION_UP:
// Allow ScrollView to intercept touch events.
scrollViewParent.requestDisallowInterceptTouchEvent(false);
return true;
case MotionEvent.ACTION_MOVE:
scrollViewParent.requestDisallowInterceptTouchEvent(true);
return false;
default:
return true;
}
}
});
答案 4 :(得分:2)
我使用了这种结构,并克服了这个问题。
我使用了地图片段的容器视图。
<ScrollView
android:layout_width="match_parent"
android:layout_height="match_parent">
<LinearLayout
android:orientation="vertical"
android:layout_width="match_parent"
android:layout_height="wrap_content">
<TextView
android:text="Another elements in scroll view1"
android:layout_width="wrap_content"
android:layout_height="wrap_content" />
<com.erolsoftware.views.MyMapFragmentContainer
android:orientation="vertical"
android:layout_width="match_parent"
android:layout_height="250dp">
<fragment
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:id="@+id/eventMap"
tools:context="com.erolsoftware.eventapp.EventDetails"
android:name="com.google.android.gms.maps.SupportMapFragment"/>
</com.erolsoftware.views.MyMapFragmentContainer>
<TextView
android:text="Another elements in scroll view2"
android:layout_width="wrap_content"
android:layout_height="wrap_content" />
</LinearLayout>
</ScrollView>
容器类:
public class MyMapFragmentContainer extends LinearLayout {
@Override
public boolean onInterceptTouchEvent(MotionEvent ev)
{
if (ev.getActionMasked() == MotionEvent.ACTION_DOWN)
{
ViewParent p = getParent();
if (p != null)
p.requestDisallowInterceptTouchEvent(true);
}
return false;
}
public MyMapFragmentContainer(Context context) {
super(context);
}
public MyMapFragmentContainer(Context context, AttributeSet attrs) {
super(context, attrs);
}
public MyMapFragmentContainer(Context context, AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
}
}
答案 5 :(得分:1)
无需为customScrollViews和透明布局文件而烦恼。只需使用较轻版本的谷歌地图,您的问题将得到解决。
map:liteMode="true"
在布局文件中的地图片段中添加上面的属性。问题将得到解决。
答案 6 :(得分:0)
对于问题的第二部分 - 您可以从SupportMapFragment派生一个片段类,并在您的布局中使用它。然后,您可以覆盖Fragment#onCreateView,并在那里实例化自定义MapView。
如果这不起作用,您可以随时创建自己的片段 - 然后您只需要自己处理所有生命周期方法(onCreate,onResume等)。 This answer有更多细节。
答案 7 :(得分:0)
<ScrollView
android:layout_width="match_parent"
::
::
::
<FrameLayout
android:id="@+id/map_add_business_one_rl"
android:layout_width="match_parent"
android:layout_height="100dp"
>
<fragment
android:id="@+id/map"
android:name="com.google.android.gms.maps.SupportMapFragment"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:layout_marginTop="-100dp"
android:layout_marginBottom="-100dp"
/>
</FrameLayout>
::
::
::
</ScrollView>