はじめに
Google-Maps-for-RailsはRailsでGoogle Mapを表示させるためのgemです。本記事では、Google Mapのマーカーに紐付いた情報ウィンドウを開くイベントを発生させる方法を紹介します。
環境
- Rails 5.1.4
- Ruby 2.4.1
- gem’gmaps4rails’(https://github.com/apneadiving/Google-Maps-for-Rails)
1. 情報ウィンドウを作成する
class MapController < ApplicationController
def index
@places =Place.all
@hash = Gmaps4rails.build_markers(@places) do |place, marker|
marker.lat place.latitude
marker.lng place.longitude
marker.infowindow render_to_string( partial: "map/infowindow",
locals: { place: place} )
marker.json({ id: place.id, })
end
end
end
<div>
<%= link_to "#{place.name}", place %>
</div>
上記のように、情報ウィンドウを作成します。イベント発生の際に、マーカーを指定できるようにするため、placeのidをマーカーに付加しておきます。
2. マーカーのserviceObjectに情報ウィンドウをセットする
マーカーのserviceObjectに、placeのidとinfowindowをセットします。
Gmaps.store = {}
Gmaps.store.markers = markers.map(function(m) {
marker = handler.addMarker(m);
marker.serviceObject.set('id', m.id);
marker.serviceObject.set('infowindow', m.infowindow);
return marker;
});
3. 情報ウィンドウのリスナーを追加する
情報ウィンドウのリスナーを追加します。イベント発生時には、指定したidのマーカーの情報ウィンドウを開くようにします。
google.maps.event.addListener(handler.getMap(), "open", function(id) {
$.each(Gmaps.store.markers, function() {
if (this.serviceObject.id == id) {
infowindow = new google.maps.InfoWindow({
content: this.serviceObject.infowindow
});
infowindow.open(this.serviceObject.map, this.serviceObject);
}
});
});
index.html.erbの全体は、以下のようになります。
<div class="map_container">
<div id="map" class="map_canvas"></div>
</div>
<script type="text/javascript">
handler = Gmaps.build('Google');
handler.buildMap({ provider: {}, internal: {id: 'map'}}, function(){
markers = <%= raw @hash.to_json %>
Gmaps.store = {}
Gmaps.store.markers = markers.map(function(m) {
marker = handler.addMarker(m);
marker.serviceObject.set('id', m.id);
marker.serviceObject.set('infowindow', m.infowindow);
return marker;
});
handler.bounds.extendWith(Gmaps.store.markers);
handler.fitMapToBounds();
handler.getMap().setCenter(new google.maps.LatLng(35.720969, 139.735877));
handler.getMap().setZoom(12);
google.maps.event.addListener(handler.getMap(), "open", function(id) {
$.each(Gmaps.store.markers, function() {
if (this.serviceObject.id == id) {
infowindow = new google.maps.InfoWindow({
content: this.serviceObject.infowindow
});
infowindow.open(this.serviceObject.map, this.serviceObject);
}
});
});
});
</script>
4. イベントを発生させる
あとは、下記でイベントを発生させることで、指定した情報ウィンドウを開くことができます。
google.maps.event.trigger(handler.getMap(), "open", id);
問題点
イベントを発生させると、マーカーをクリックしたときに、二重に情報ウィンドウを開いてしまう。
解決法
以下のコードを追記することで、クリックイベントで開いた情報ウィンドウを閉じることが出来ます。
if (handler.currentInfowindow()) {
handler.currentInfowindow().close();
}
まとめ
Google-Maps-for-Railsを使用して、Google Mapのマーカーに紐付いた情報ウィンドウを開くイベントを発生させる方法を紹介しました。この方法を利用して、マーカーのクリックイベント以外でも、好きなタイミングで指定したマーカーの情報ウィンドウを開くことができます。