実現したいこと
MapKit
の地図にダークモードを適用させたいです。
MapKit
を使って一定の縮尺に達するとMapTypeを切り替えたいです。
Google Mapでは似たようなことが出来るのでAppleのMapKitで実現出来ないか知りたいです。
https://note.com/runomimi/n/n3dc7cdcb6a62
前提
・AppleのMapKit
を使用しています。
・ダークモード表示は成功しました。
・デフォルトは .hybridFlyover
を使用しています。
・ある一定のレベル
(例:日本地図の拡大図→東京をタップ→東京の縮尺が表示)
されると地図を .standard
にして ダークモード
に切り替えたいです。
(UI/UXの都合上です。)
ダークモード表示に成功したので、今度はカスタマイズに挑戦したところ何故か ダークモード
が解除されてしまいました。
発生している問題・エラーメッセージ
ダークモードに切り替わらない
該当のソースコード
Swift
1// ダークモード成功 2// 参考:https://stackoverflow.com/questions/58565004/how-can-i-turn-the-mkmapview-for-dark-mode 3 4override func viewDidLoad() { 5 super.viewDidLoad() 6 7mapView.mapType = .standard 8// ダークモード 9if #available(iOS 13.0, *) { 10 self.overrideUserInterfaceStyle = .dark 11} 12}
Swift
1// ダークモード失敗 2// カスタマイズ時 3 4import UIKit 5import MapKit 6import CoreLocation 7 8class TestMapKitViewController: UIViewController, MKMapViewDelegate, CLLocationManagerDelegate { 9 10 // 接続 11 @IBOutlet weak var mapView: MKMapView! 12 13 // 変数 14 // 地図に立てるピン 15 var locationManager: CLLocationManager! 16 17 override func viewDidLoad() { 18 super.viewDidLoad() 19 20 // 非同期開始 21 DispatchQueue.main.asyncAfter(deadline: DispatchTime.now() + 1) { 22 self.focusOnPointAndShowGlobe() 23 } 24 25 // 縮尺を設定 26 var region:MKCoordinateRegion = mapView.region 27 region.span.latitudeDelta = 0.02 28 region.span.longitudeDelta = 0.02 29 // セット 30 mapView.setRegion(region,animated:true) 31 32 // デリゲート 33 mapView.delegate = self 34 35 // 表示タイプ 36 self.mapView.mapType = .hybridFlyover 37 38 // ダークモード 39 if #available(iOS 13.0, *) { 40 self.overrideUserInterfaceStyle = .dark 41 } 42 43 // 今回は現在地とする 44 let coordinate = mapView.userLocation.coordinate 45 // ピンを生成 46 let pin = MKPointAnnotation() 47 // ピンのタイトル・サブタイトルをセット 48 pin.title = "タイトル" // 緯度・経度を入れる 49 pin.subtitle = "サブタイトル" 50 // ピンに一番上で作った位置情報をセット 51 pin.coordinate = coordinate 52 // mapにピンを表示する 53 mapView.addAnnotation(pin) 54 55 // ロングタップを検知 56 let longPress = UILongPressGestureRecognizer(target: self, action: #selector(recognizeLongPress(sender:))) 57 //MapViewにリスナーを登録 58 self.mapView.addGestureRecognizer(longPress) 59 } 60 61 func focusOnPointAndShowGlobe() { 62 let mapHeight = mapView.bounds.height // the constraint of visible Map area will base on HEIGHT of mapview 63 64 let standardMapHeight: CGFloat = 450 65 let standardCameraDistance: CGFloat = 24 * 1000 * 1000 // 24,000 km 66 67 let targetDistance = (mapHeight / standardMapHeight) * standardCameraDistance 68 let center = CLLocationCoordinate2D(latitude: 緯度, longitude: 経度) 69 let camera = MKMapCamera(lookingAtCenter: center, fromDistance: targetDistance, pitch: 0, heading: 0) 70 self.mapView.setCamera(camera, animated: true) 71 } 72 73 func mapView(_ mapView: MKMapView, didSelect view: MKAnnotationView) { 74 // タップされたピンの位置情報 75 print(view.annotation?.coordinate as Any) 76 // タップされたピンのタイトルとサブタイトル 77 print(view.annotation?.title!! ?? "title") 78 print(view.annotation?.subtitle) 79 } 80 81 //ロングタップした時に呼ばれる関数 82 @objc func recognizeLongPress(sender: UILongPressGestureRecognizer) { 83 //長押し感知は最初の1回のみ 84 if sender.state != UIGestureRecognizer.State.began { 85 return 86 } 87 88 // 位置情報を取得 89 let location = sender.location(in: self.mapView) 90 let coordinate = self.mapView.convert(location, toCoordinateFrom: self.mapView) 91 92 // 緯度取得 93 self.longitude = String(coordinate.latitude) 94 // 出力 95 print(coordinate.latitude) 96 97 // 経度取得 98 self.latitude = String(coordinate.longitude) 99 // 出力 100 print(coordinate.longitude) 101 102 // タップした位置に照準を合わせる処理 103 let span = MKCoordinateSpan(latitudeDelta: 0.01, longitudeDelta: 0.01) 104 let region = MKCoordinateRegion(center: coordinate, span: span) 105 self.mapView.region = region 106 107 // ピンを生成 108 let pin = MKPointAnnotation() 109 110 pin.subtitle = "緯度:" + self.longitude + "経度:" + self.latitude 111 // タップした位置情報に位置にピンを追加 112 pin.coordinate = coordinate 113 self.mapView.addAnnotation(pin) 114 } 115 116 //アノテーションビューを返すメソッド 117 func mapView(_ mapView: MKMapView, viewFor annotation: MKAnnotation) -> MKAnnotationView? { 118 // MapViewのdelegateメソッド、viewFor annotationを定義している場合、 119 // 地図上に青点のannotationを表示する処理がオーバライドされてしまう。 120 // 青点のannotation(現在地表示しているピン)のみ、処理しないという記述を行うことで正しい動きになる。 121 if annotation.title == "My Location"{ 122 return nil 123 } 124 125 //アノテーションビューの作成 126 let pinView = MKMarkerAnnotationView(annotation: annotation, reuseIdentifier: nil) 127 128 //吹き出しを表示可能にする 129 pinView.canShowCallout = true 130 131 // 例で削除ボタンの作成 132 let button = UIButton() 133 button.frame = CGRect(x:0,y:0,width:40,height:40) 134 button.setImage(UIImage(systemName: "trash"), for: .normal) 135 button.tintColor = UIColor.red 136 137 //右側に削除ボタンを追加 138 pinView.rightCalloutAccessoryView = button 139 140 return pinView 141 } 142 143 func mapView(_ mapView: MKMapView, annotationView view: MKAnnotationView, calloutAccessoryControlTapped control: UIControl) { 144 guard let annotation = view.annotation else { 145 return 146 } 147 //右のボタンが押された場合はピンを消す。 148 self.mapView.removeAnnotation(annotation) 149 } 150} 151 152
調べていること
・表示領域の変化イベントを取得する方法
https://teratail.com/questions/138369
・地図上の縮小・拡大に関するコード
https://qiita.com/shinobee/items/4406c4fdf2fbdcc40533
試したこと
・表示領域の変化を取得→一定の縮尺に達するとMapTypeを切り替える流れで実装できるのか知りたいです。
(同じような処理をしている例があまり見当たらなかったです。
調べているのですが地図のカスタマイズに疎く初心者でよくわかりません。
お手柔らかにお願いします。
回答1件
あなたの回答
tips
プレビュー