地図の上にタイル レイヤとして画像を追加できます。タイルレイヤは、特定のズームレベルで地図タイルの上に配置されます。十分な数のタイルがあれば、複数のズームレベルでマップ全体の Google マップデータを補うことができます。
はじめに
タイルレイヤ(タイル オーバーレイとも呼ばれる)を使用すると、Google の基本地図タイルの上に画像を重ねることができます。これは、スポットや交通情報などのデータやローカル画像をアプリに追加する優れた方法です。kGMSTypeNone
地図タイプと組み合わせると、タイルレイヤを使用して Google の基本地図データを独自のデータに置き換えることができます。
タイルレイヤは、広い地理的領域をカバーする画像をマップに追加する場合に役立ちます。地面オーバーレイは、地図上の 1 つのポイントに 1 つの画像を固定したい場合に役立ちます。
タイル座標
Maps API は、各ズームレベルの画像を、グリッドに配置された一連の正方形のマップタイルに分割します。地図を新しい位置にスクロールする、または新しいズームレベルにスクロールすると、Maps API は必要なタイルを決定し、そのタイルを取得するタイルのグループに変換します。
Google によるメルカトル図法の実装では、座標(0,0)のタイルは常に地図の北西端になり、x
値は西から東に増分し、y
値は北から南に増分します。タイルは、この原点からの x,y
座標を使用してインデックスが付けられます。たとえば、ズームレベル 2 で地球が 16 個のタイルに分割された場合は、各タイルは一意の x,y
ペアで参照できます。
各マップ タイプは 256 x 256 ポイントの正方形です。ズームレベル 0 では、全世界が 1 つのタイルでレンダリングされます。各ズームレベルは、2 の倍数で倍率が増大します。したがって、ズームレベルが 1 の場合、マップは 2x2 グリッドのタイルとしてレンダリングされ、ズームレベルが 2 の場合は 4x4 グリッド、ズームレベルが 3 の場合は 8x8 グリッド、というように続きます。タイルレイヤの画像を作成する場合は、サポートするズームレベルごとに、各タイルの 256 x 256 ポイントの新しい画像を作成する必要があります。
タイル レイヤの追加
GMSURLTileLayer
オブジェクト、またはGMSTileLayer
またはGMSSyncTileLayer
のカスタム サブクラスをインスタンス化します。- 必要に応じて
zIndex
プロパティを変更して、他のタイルレイヤとの位置関係を調整します。 map
プロパティを設定して、GMSTileLayer
オブジェクトを地図に割り当てます。
Maps SDK for iOS には、タイルレイヤの実装に使用できる 3 つのクラスが用意されています。各クラスで、特定の {x,y,zoom}
座標セットに適した地図タイルを取得する方法を定義する必要があります。使用できるオプションは次のとおりです。
GMSSyncTileLayer
のサブクラスを作成し、UIImage
インスタンスを返すtileForX:y:zoom
の実装を提供します。GMSTileLayer
のサブクラスを作成し、後でタイル画像でコールバックする非同期メソッドrequestTileForX:y:zoom
の実装を提供します。- 既存のクラス
GMSURLTileLayer
を使用して、URL からタイルを自動的に取得し、GMSTileURLConstructor
ブロックを提供します。GMSURLTileLayer
は、サブクラス化できない具象クラスです。
GMSSyncTileLayer
または GMSTileLayer
のサブクラス化の場合、nil
タイルの結果を指定すると、現在はデータが利用できないが、将来的には利用可能になる可能性があることを Maps SDK for iOS に伝えます。または、この場所にタイルがないことを示すために kGMSTileLayerNoTile
を返します。
GMSURLTileLayer
の場合、GMSTileURLConstructor
から nil
が返された場合は、この場所にタイルがないことを示します。
GMSURLTileLayer を使用して URL からタイルを取得
GMSURLTileLayer
ではサブクラス化は必要ありませんが、GMSTileURLConstructor
ブロックを実装する必要があります。次のコードは、GMSURLTileLayer
を使用して複数階建ての建物のフロアプランを表示する方法を示しています。
Swift
let floor = 1 // Implement GMSTileURLConstructor // Returns a Tile based on the x,y,zoom coordinates, and the requested floor let urls: GMSTileURLConstructor = { (x, y, zoom) in let url = "https://www.example.com/floorplans/L\(floor)_\(zoom)_\(x)_\(y).png" return URL(string: url) } // Create the GMSTileLayer let layer = GMSURLTileLayer(urlConstructor: urls) // Display on the map at a specific zIndex layer.zIndex = 100 layer.map = mapView
Objective-C
NSInteger floor = 1; // Create the GMSTileLayer GMSURLTileLayer *layer = [GMSURLTileLayer tileLayerWithURLConstructor:^NSURL * _Nullable(NSUInteger x, NSUInteger y, NSUInteger zoom) { NSString *url = [NSString stringWithFormat:@"https://www.example.com/floorplans/L%ld_%lu_%lu_%lu.png", (long)floor, (unsigned long)zoom, (unsigned long)x, (unsigned long)y]; return [NSURL URLWithString:url]; }]; // Display on the map at a specific zIndex layer.zIndex = 100; layer.map = mapView;
GMSSyncTileLayer をサブクラス化して UIImage としてタイルを提供
GMSSyncTileLayer
と GMSTileLayer
は、サブクラス化するように設計された抽象クラスです。これらのクラスを使用して、タイルを UIImage
として提供できます。次の例は、GMSSyncTileLayer
をサブクラス化することで、地図上の一部のタイルにカスタム画像をレンダリングする方法を示しています。
Swift
class TestTileLayer: GMSSyncTileLayer { override func tileFor(x: UInt, y: UInt, zoom: UInt) -> UIImage? { // On every odd tile, render an image. if (x % 2 == 1) { return UIImage(named: "australia") } else { return kGMSTileLayerNoTile } } }
Objective-C
@interface TestTileLayer : GMSSyncTileLayer @end @implementation TestTileLayer - (UIImage *)tileForX:(NSUInteger)x y:(NSUInteger)y zoom:(NSUInteger)zoom { // On every odd tile, render an image. if (x % 2 == 1) { return [UIImage imageNamed:@"australia"]; } else { return kGMSTileLayerNoTile; } } @end
マップにレイヤーを追加するには、オブジェクトをインスタンス化して map プロパティを設定します。
Swift
let layer = TestTileLayer() layer.map = mapView
Objective-C
GMSTileLayer *layer = [[TestTileLayer alloc] init]; layer.map = mapView;
Retina 端末用の高 DPI タイル
tileSize
を 512 に設定すると、GMSSyncTileLayer
または GMSURLTileLayer
で高 DPI 画像を使用できます。tileSize
プロパティは、返されたタイル画像の表示に使用するピクセル数を示します。デフォルトは 256 で、Retina 以外のデバイスの Google マップ タイルのサイズです。
高 DPI デバイスに通常の DPI タイルを表示する場合は、tileSize
を 512 に設定して画像を拡大できます。画像をアップスケールすると、特に細い線やテキストの画質が低下する可能性があります。最適な結果を得るには、tileSize
と画像の DPI をディスプレイに合わせてください。Retina デバイスで表示される地図は、tileSize
が 512 の HiDPI 画像を表示すると見栄えがよく、Retina 以外のデバイスで表示される地図は、通常の画像とデフォルトの tileSize
256 で見栄えがよく、
古いタイルの消去
レイヤによって提供されたタイルが「古く」なったら、レイヤでメソッド clearTileCache
を呼び出して強制的に更新する必要があります。これにより、このレイヤ上のすべてのタイルがリロードされます。
Swift
layer.clearTileCache()
Objective-C
[layer clearTileCache];