ユーザーがGoogleマップアクティビティを開いたらすぐにスナックバーを表示したいのですが、アクティビティの最初のパラメーターとして(findViewById()
のSnackbar.make()
で)使用するビューがアクティビティにないということです。そこに何を置きますか? Javaクラスコード:
public class MapsActivity extends FragmentActivity implements OnMapReadyCallback {
private GoogleMap mMap;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_maps);
// Obtain the SupportMapFragment and get notified when the map is ready to be used.
SupportMapFragment mapFragment = (SupportMapFragment) getSupportFragmentManager()
.findFragmentById(R.id.map);
mapFragment.getMapAsync(this);
}
@Override
public void onMapReady(GoogleMap googleMap) {
mMap = googleMap;
mMap.setBuildingsEnabled(true);
mMap.getUiSettings().setZoomControlsEnabled(true);
float cameraZoom = 17;
LatLng location = new LatLng(43.404032, -80.478184);
mMap.addMarker(new MarkerOptions().position(location).title("49 McIntyre Place #18, Kitchener, ON N2R 1G3"));
CameraUpdateFactory.newLatLngZoom(location, cameraZoom);
Snackbar.make(findViewById(/*WHAT DO I PUT HERE?*/), "Click the pin for more options", Snackbar.LENGTH_LONG).show();
}
}
また、アクティビティxmlコードは次のとおりです。
<fragment xmlns:Android="http://schemas.Android.com/apk/res/Android"
xmlns:map="http://schemas.Android.com/apk/res-auto"
xmlns:tools="http://schemas.Android.com/tools"
Android:id="@+id/map"
Android:name="com.google.Android.gms.maps.SupportMapFragment"
Android:layout_width="match_parent"
Android:layout_height="match_parent"
tools:context="ca.davesautoservice.davesautoservice.MapsActivity" />
最後に、スタックトレースエラーを示します。
08-03 11:42:21.333 3901-3901/? E/AndroidRuntime: FATAL EXCEPTION: main
Process: ca.davesautoservice.davesautoservice, PID: 3901
Java.lang.NullPointerException
at Android.support.design.widget.Snackbar.<init>(Snackbar.Java:183)
at Android.support.design.widget.Snackbar.make(Snackbar.Java:215)
at ca.davesautoservice.davesautoservice.MapsActivity.onMapReady(MapsActivity.Java:48)
at com.google.Android.gms.maps.SupportMapFragment$zza$1.zza(Unknown Source)
at com.google.Android.gms.maps.internal.zzo$zza.onTransact(Unknown Source)
at Android.os.Binder.transact(Binder.Java:361)
at xz.a(:com.google.Android.gms.DynamiteModulesB:82)
at maps.ad.u$5.run(Unknown Source)
at Android.os.Handler.handleCallback(Handler.Java:808)
at Android.os.Handler.dispatchMessage(Handler.Java:103)
at Android.os.Looper.loop(Looper.Java:193)
at Android.app.ActivityThread.main(ActivityThread.Java:5333)
at Java.lang.reflect.Method.invokeNative(Native Method)
at Java.lang.reflect.Method.invoke(Method.Java:515)
at com.Android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.Java:828)
at com.Android.internal.os.ZygoteInit.main(ZygoteInit.Java:644)
at dalvik.system.NativeStart.main(Native Method)
助けてくれてありがとう! :)
いくつかのオプションが表示されます...どれが問題を解決できるかわかりません。
最も単純な
SupportMapFragment
はクラス_Android.support.v4.app.Fragment
_を拡張します。このように、メソッドにはgetView()
があります
_Snackbar.make(mapFragment.getView(), "Click the pin for more options", Snackbar.LENGTH_LONG).show();
_
ルートビューを検索
この回答から 、ルートビューを取得する方法があります:
_getWindow().getDecorView().getRootView()
_
だから、多分、あなたはすることができます:
_Snackbar.make(getWindow().getDecorView().getRootView(), "Click the pin for more options", Snackbar.LENGTH_LONG).show();
_
ビューを取得するためにダミーのLinearLayoutを追加します
正直なところ、この解決策が可能かどうかはわかりません。 Mapsフラグメントの上にLinearLayoutを追加できるかどうかはわかりません...それでいいと思いますが、以前Maps APIを使用したことがないので、わかりません。
_<LinearLayout xmlns:Android="http://schemas.Android.com/apk/res/Android"
Android:id="@+id/dummy_layout_for_snackbar"
Android:layout_width="match_parent"
Android:layout_height="match_parent"
Android:orientation="vertical">
<fragment
xmlns:map="http://schemas.Android.com/apk/res-auto"
xmlns:tools="http://schemas.Android.com/tools"
Android:id="@+id/map"
Android:name="com.google.Android.gms.maps.SupportMapFragment"
Android:layout_width="match_parent"
Android:layout_height="match_parent"
tools:context="ca.davesautoservice.davesautoservice.MapsActivity" />
</LinearLayout>
_
その後:
_Snackbar.make(findViewById(R.id.dummy_layout_for_snackbar), "Click the pin for more options", Snackbar.LENGTH_LONG).show();
_
Shahab Raufが指摘しているように、getDecorView()を介してビューを取得すると、下部のナビゲーションバーの後ろにスナックバーが配置される場合があります。私は次のコードを使用します:(あなたの喜びに使用して拡張します)
public class BaseActivity extends AppCompatActivity {
private View getRootView() {
final ViewGroup contentViewGroup = (ViewGroup) findViewById(Android.R.id.content);
View rootView = null;
if(contentViewGroup != null)
rootView = contentViewGroup.getChildAt(0);
if(rootView == null)
rootView = getWindow().getDecorView().getRootView();
return rootView;
}
protected void showSnackBarWithOK(@StringRes int res) {
final View rootView = getRootView();
if(rootView != null) {
final Snackbar snackbar = Snackbar.make(getRootView(), res, Snackbar.LENGTH_INDEFINITE);
snackbar.setAction(R.string.ok, new View.OnClickListener() {
@Override
public void onClick(View v) {
snackbar.dismiss();
}
});
snackbar.show();
}
}
protected void showSnackBar(@StringRes int res) {
final View rootView = getRootView();
if(rootView != null)
Snackbar.make(rootView, res, Snackbar.LENGTH_LONG).show();
}
}
findViewById
が最良のアプローチだと思います。 getWindow().getDecorView().getRootView()
を使用すると、下部のナビゲーションバーの背後にメッセージが表示されます。
私の場合、XMLでダミー要素を作成する必要はありませんでした。私は次のコードを使用しました
Snackbar.make(findViewById(R.id.container),"This will start payment process.", Snackbar.LENGTH_LONG)
.setAction("Action", null).show();
私のXMLファイルは次のとおりです
<Android.support.design.widget.CoordinatorLayout
xmlns:Android="http://schemas.Android.com/apk/res/Android"
xmlns:app="http://schemas.Android.com/apk/res-auto"
xmlns:tools="http://schemas.Android.com/tools"
Android:layout_width="match_parent"
Android:layout_height="match_parent"
Android:fitsSystemWindows="true"
tools:context=".HomePatientActivity">
<Android.support.design.widget.AppBarLayout
Android:layout_width="match_parent"
Android:layout_height="wrap_content"
Android:paddingTop="@dimen/appbar_padding_top"
Android:theme="@style/AppTheme.AppBarOverlay">
<Android.support.v7.widget.Toolbar
Android:id="@+id/toolbar"
Android:layout_width="match_parent"
Android:layout_height="?attr/actionBarSize"
Android:layout_weight="1"
Android:background="?attr/colorPrimary"
app:layout_scrollFlags="scroll|enterAlways"
app:popupTheme="@style/AppTheme.PopupOverlay">
</Android.support.v7.widget.Toolbar>
<Android.support.design.widget.TabLayout
Android:id="@+id/sliding_tabs"
Android:layout_width="match_parent"
Android:layout_height="match_parent"
Android:layout_weight="1"
app:layout_anchorGravity="top"
app:tabMode="fixed" />
</Android.support.design.widget.AppBarLayout>
<Android.support.v4.view.ViewPager
Android:id="@+id/container"
Android:layout_width="match_parent"
Android:layout_height="wrap_content"
app:layout_behavior="@string/appbar_scrolling_view_behavior">
</Android.support.v4.view.ViewPager>
<Android.support.design.widget.FloatingActionButton
Android:id="@+id/fab"
Android:layout_width="match_parent"
Android:layout_height="wrap_content"
Android:layout_gravity="end|bottom"
Android:layout_margin="@dimen/fab_margin"
app:srcCompat="@Android:drawable/ic_dialog_email" />
</Android.support.design.widget.CoordinatorLayout>
レイアウトを追加し、以下のidの例をレイアウトに割り当てます。
<RelativeLayout
Android:id="@+id/relativeLayout"
Android:layout_width="match_parent"
Android:layout_height="match_parent">
<fragment
Android:id="@+id/map_fragment"
Android:layout_width="match_parent"
Android:layout_height="match_parent"
Android:name="com.google.Android.gms.maps.SupportMapFragment"
/>
</RelativeLayout>
アクティビティで、findViewByIdを使用してレイアウトを見つけます。
RelativeLayout relativeLayout = findViewById(R.id.relativeLayout);
レイアウトにIDを割り当てる限り、それぞれ他のレイアウトの例:
LinearLayout relativeLayout = findViewById(R.id.linearLayout);
FrameLayout frameLayout = findViewById(R.id.frameLayout);
スナックバーの表示:
Snackbar.make(relativeLayout,"You are displaying a Snackbar",Snackbar.LENGTH_SHORT).show();
他のレイアウトの場合は、レイアウトの例の名前を変更するだけです:
Snackbar.make(linearLayout,"You are displaying a Snackbar",Snackbar.LENGTH_SHORT).show();
Snackbar.make(frameLayout,"You are displaying a Snackbar",Snackbar.LENGTH_SHORT).show();
任意のアクティビティでこれを試してください:
snackbar(findViewById(Android.R.id.content),"your text")