axiom-mapkit-diag
SKILL.md
MapKit Diagnostics
Symptom-based MapKit troubleshooting. Start with the symptom you're seeing, follow the diagnostic path.
Related Skills
axiom-mapkit— Patterns, decision trees, anti-patternsaxiom-mapkit-ref— API reference, code examples
Quick Reference
| Symptom | Check First | Common Fix |
|---|---|---|
| Annotations not appearing | Coordinate values (lat/lng swapped?) | Verify coordinate, check viewFor delegate |
| Map region jumps/loops | updateUIView guard | Add region equality check |
| Slow with many annotations | Annotation count, view reuse | Enable clustering, implement view reuse |
| Clustering not working | clusteringIdentifier set? | Set same identifier on all views |
| Overlays not rendering | renderer delegate method | Return correct MKOverlayRenderer subclass |
| Search returns no results | resultTypes, region bias | Set appropriate resultTypes and region |
| User location not showing | Authorization status | Request CLLocationManager authorization first |
| Coordinates appear wrong | lat/lng order | MapKit uses (latitude, longitude) — verify data source |
Symptom 1: Annotations Not Appearing
Decision Tree
Q1: Are coordinates valid?
├─ 0,0 or NaN → Data source returning default/empty values
│ Fix: Validate coordinates before adding annotations
│ Debug: print("\(annotation.coordinate.latitude), \(annotation.coordinate.longitude)")
│
└─ Valid numbers → Check next
Q2: Are lat/lng swapped?
├─ YES (common with GeoJSON which uses [longitude, latitude]) → Swap values
│ GeoJSON: [lng, lat] — MapKit: CLLocationCoordinate2D(latitude:, longitude:)
│ Fix: CLLocationCoordinate2D(latitude: json[1], longitude: json[0])
│
└─ NO → Check next
Q3: (MKMapView) Is mapView(_:viewFor:) delegate returning nil for your annotations?
├─ Not implemented → System uses default pin (should appear)
├─ Returns nil → System uses default pin (should appear)
├─ Returns wrong view → Check implementation
│
└─ Check delegate is set
Q4: (MKMapView) Is delegate set?
├─ NO → mapView.delegate = self (or context.coordinator in UIViewRepresentable)
│ Without delegate: default pins appear. But if viewFor returns nil, check annotation type
│
└─ YES → Check next
Q5: (SwiftUI) Are annotations in Map content builder?
├─ NO → Annotations must be inside Map { ... } content closure
│ Fix: Map(position: $pos) { Marker("Name", coordinate: coord) }
│
└─ YES → Check next
Q6: Is the map region showing the annotation coordinates?
├─ Map centered elsewhere → Adjust camera/region to include annotation coordinates
│ Debug: Compare mapView.region with annotation coordinates
│ Fix: Use .automatic camera position or set region to fit annotations
│
└─ Region includes annotations → Check displayPriority
Q7: (MKMapView) Is displayPriority too low?
├─ .defaultLow → System may hide annotations at certain zoom levels
│ Fix: view.displayPriority = .required for must-show annotations
│
└─ .required → Annotation should appear — file a bug report with minimal repro
Symptom 2: Map Region Jumping / Infinite Loops
Decision Tree
Q1: (UIViewRepresentable) Is setRegion called in updateUIView without guard?
├─ YES → Classic infinite loop:
│ 1. SwiftUI state changes → updateUIView called
│ 2. updateUIView calls setRegion
│ 3. setRegion triggers regionDidChangeAnimated delegate
│ 4. Delegate updates SwiftUI state → back to step 1
│
│ Fix: Guard against unnecessary updates
│ if mapView.region.center.latitude != region.center.latitude
│ || mapView.region.center.longitude != region.center.longitude {
│ mapView.setRegion(region, animated: true)
│ }
│
│ Alternative: Use a flag in coordinator
│ coordinator.isUpdating = true
│ mapView.setRegion(region, animated: true)
│ coordinator.isUpdating = false
│ // In regionDidChangeAnimated: guard !isUpdating
│
└─ NO → Check next
Q2: Are multiple state sources fighting over the region?
├─ YES → Two bindings or state variables controlling the same region
│ Fix: Single source of truth for camera position
│ One @State var cameraPosition, not two conflicting values
│
└─ NO → Check next
Q3: (SwiftUI) Is MapCameraPosition properly bound?
├─ Using .constant() or recreating position on each render → Camera resets
│ Fix: @State private var cameraPosition: MapCameraPosition = .automatic
│ Use the binding: Map(position: $cameraPosition)
│
└─ Properly bound → Check next
Q4: Animation conflict?
├─ Using animated: true in updateUIView alongside SwiftUI animations → Double animation
│ Fix: Avoid animated: true in updateUIView, or disable SwiftUI animation for map
│
└─ NO → Check next
Q5: Is onMapCameraChange triggering state updates that move the camera?
├─ YES → Camera change → callback → state change → camera change
│ Fix: Only update non-camera state in the callback
│ Don't set cameraPosition inside onMapCameraChange
│
└─ NO → Check delegate implementation for unintended state mutations
Symptom 3: Performance Issues
Decision Tree
Q1: How many annotations?
├─ > 500 without clustering → Enable clustering
│ SwiftUI: .mapItemClusteringIdentifier("poi")
│ MKMapView: view.clusteringIdentifier = "poi"
│
├─ > 1000 → Consider visible-region filtering
│ Only load annotations within mapView.region
│ Use .onMapCameraChange to fetch when user scrolls
│
└─ < 500 → Check next
Q2: (MKMapView) Using dequeueReusableAnnotationView?
├─ NO → Every annotation creates a new view → memory spike
│ Fix: Register view class and dequeue in delegate
│ mapView.register(MKMarkerAnnotationView.self, forAnnotationViewWithReuseIdentifier: "marker")
│
└─ YES → Check next
Q3: Complex custom annotation views?
├─ YES → Rich SwiftUI views or complex UIViews per annotation
│ Fix: Pre-render to UIImage for MKAnnotationView.image
│ Or simplify to MKMarkerAnnotationView with glyph
│
└─ NO → Check next
Q4: Overlays with many coordinates?
├─ YES → Polylines/polygons with 10K+ points
│ Fix: Simplify geometry (Douglas-Peucker algorithm)
│ Or render at reduced detail for zoomed-out views
│
└─ NO → Check next
Q5: Geocoding in a loop?
├─ YES → CLGeocoder has rate limit (~1/second)
│ Fix: Batch geocoding, throttle requests, cache results
│ Use MKLocalSearch for batch lookups instead of per-item geocoding
│
└─ NO → Profile with Instruments → Time Profiler for CPU, Allocations for memory
Symptom 4: Clustering Not Working
Decision Tree
Q1: Is clusteringIdentifier set on annotation views?
├─ NO → Clustering requires an identifier on each annotation view
│ MKMapView: view.clusteringIdentifier = "poi" in viewFor delegate
│ SwiftUI: .mapItemClusteringIdentifier("poi") on content
│
└─ YES → Check next
Q2: Are ALL relevant views using the SAME identifier?
├─ NO → Different identifiers = different cluster groups
│ Fix: Use consistent identifier for annotations that should cluster together
│
└─ YES → Check next
Q3: (MKMapView) Is mapView(_:clusterAnnotationForMemberAnnotations:) needed?
├─ Not implemented → System creates default cluster
│ If you need custom cluster appearance, implement this delegate method
│
└─ Implemented → Check return value
Q4: Too few annotations in visible area?
├─ YES → Clustering only activates when annotations physically overlap
│ At low zoom (city level), 10 annotations might cluster
│ At high zoom (street level), same 10 might all be visible individually
│
└─ NO → Check next
Q5: (MKMapView) Are annotation views registered?
├─ NO → Register both individual and cluster view classes
│ mapView.register(MKMarkerAnnotationView.self, forAnnotationViewWithReuseIdentifier: "marker")
│
└─ YES → Verify viewFor delegate handles both MKClusterAnnotation and individual annotations
Symptom 5: Overlays Not Rendering
Decision Tree
Q1: (MKMapView) Is mapView(_:rendererFor:) delegate method implemented?
├─ NO → Overlays require a renderer — without this delegate method, nothing renders
│ Fix: Implement the delegate method, return appropriate renderer subclass
│
└─ YES → Check next
Q2: Is the correct renderer subclass returned?
├─ MKCircle → MKCircleRenderer
│ MKPolyline → MKPolylineRenderer
│ MKPolygon → MKPolygonRenderer
│ MKTileOverlay → MKTileOverlayRenderer
│ Mismatch → Crash or silent failure
│
└─ Correct → Check next
Q3: Is renderer styled?
├─ No strokeColor/fillColor/lineWidth set → Renderer exists but invisible
│ Fix: Set at minimum strokeColor and lineWidth
│ renderer.strokeColor = .systemBlue
│ renderer.lineWidth = 2
│
└─ Styled → Check next
Q4: Overlay level wrong?
├─ .aboveRoads → Overlay may be behind labels (hard to see)
│ Try: mapView.addOverlay(overlay, level: .aboveLabels)
│
└─ Check overlay coordinates match visible region
Q5: (SwiftUI) Using MapCircle/MapPolyline without styling?
├─ No .foregroundStyle or .stroke → May render transparent
│ Fix: MapCircle(center: coord, radius: 500)
│ .foregroundStyle(.blue.opacity(0.3))
│ .stroke(.blue, lineWidth: 2)
│
└─ Styled → Check coordinates are within visible map region
Symptom 6: Search / Directions Failures
Decision Tree
Q1: Network available?
├─ NO → MapKit search requires network connectivity
│ Fix: Check URLSession connectivity or NWPathMonitor
│
└─ YES → Check next
Q2: resultTypes too restrictive?
├─ Only .physicalFeature but searching for "Starbucks" → No results
│ Fix: Use .pointOfInterest for businesses, .address for streets
│ Or combine: [.pointOfInterest, .address]
│
└─ Appropriate → Check next
Q3: Region bias missing?
├─ NO region set → Results may be from anywhere in the world
│ Fix: request.region = mapView.region (or visible region)
│ This biases results to what the user can see
│
└─ Region set → Check next
Q4: Natural language query format?
├─ Structured format (lat/lng, codes) → Won't parse
│ Good: "coffee shops near San Francisco"
│ Good: "123 Main St"
│ Bad: "lat:37.7 lng:-122.4 coffee"
│ Bad: "POI_TYPE=cafe"
│
└─ Natural language → Check next
Q5: Rate limited?
├─ Getting errors after many requests → Apple rate-limits MapKit search
│ Fix: Throttle searches, use MKLocalSearchCompleter for autocomplete
│ Don't fire MKLocalSearch on every keystroke
│
└─ NO → Check next
Q6: (Directions) Source and destination valid?
├─ source or destination is nil → Request will fail
│ Fix: Verify both are valid MKMapItem instances
│ MKMapItem.forCurrentLocation() requires location authorization
│
└─ Both valid → Check transportType availability
Transit directions not available in all regions
Walking/driving available globally
Symptom 7: User Location Not Showing
Decision Tree
Q1: What is CLLocationManager.authorizationStatus?
├─ .notDetermined → Authorization never requested
│ Fix: Request authorization first, then enable user location
│ CLServiceSession(authorization: .whenInUse)
│
├─ .denied → User denied location access
│ Fix: Show UI explaining value, link to Settings
│
├─ .restricted → Parental controls block access
│ Fix: Inform user, cannot override
│
└─ .authorizedWhenInUse / .authorizedAlways → Check next
Q2: (MKMapView) Is showsUserLocation set to true?
├─ NO → mapView.showsUserLocation = true
│
└─ YES → Check next
Q3: (SwiftUI) Using UserAnnotation() in Map content?
├─ NO → Add UserAnnotation() inside Map { ... }
│
└─ YES → Check next
Q4: Running in Simulator?
├─ YES, no custom location set → Simulator doesn't have GPS
│ Fix: Debug menu → Location → Custom Location (or Apple/City Bicycle Ride/etc.)
│ Xcode: Debug → Simulate Location → pick a location
│
└─ Physical device → Check next
Q5: MapKit implicitly requests authorization — was it previously denied?
├─ MapKit shows no prompt if already denied
│ Check: Settings → Privacy & Security → Location Services → Your App
│ If "Never": User must manually re-enable
│
└─ Authorized → Check if location services enabled system-wide
Settings → Privacy & Security → Location Services → toggle at top
Q6: Location icon appearing but blue dot not on screen?
├─ User is outside the visible map region
│ Fix: Use MapCameraPosition.userLocation(fallback: .automatic)
│ Or add MapUserLocationButton() in .mapControls
│
└─ See axiom-core-location-diag for deeper location troubleshooting
Symptom 8: Coordinate System Confusion
Common coordinate mistakes that cause annotations to appear in wrong locations.
MapKit vs GeoJSON
| System | Order | Example |
|---|---|---|
| MapKit (CLLocationCoordinate2D) | latitude, longitude | CLLocationCoordinate2D(latitude: 37.77, longitude: -122.42) |
| GeoJSON | longitude, latitude | [-122.42, 37.77] |
| Google Maps | latitude, longitude | Same as MapKit |
| PostGIS ST_MakePoint | longitude, latitude | Same as GeoJSON |
The #1 coordinate bug: Swapping lat/lng when parsing GeoJSON.
// ❌ WRONG: Using GeoJSON order directly
let coord = CLLocationCoordinate2D(
latitude: geoJson[0], // This is longitude!
longitude: geoJson[1] // This is latitude!
)
// ✅ RIGHT: GeoJSON is [lng, lat], MapKit wants (lat, lng)
let coord = CLLocationCoordinate2D(
latitude: geoJson[1],
longitude: geoJson[0]
)
MKMapPoint vs CLLocationCoordinate2D
CLLocationCoordinate2D— geographic coordinates (lat/lng in degrees)MKMapPoint— projected coordinates for flat map rendering- Convert:
MKMapPoint(coordinate)andcoordinateproperty on MKMapPoint - Never use MKMapPoint x/y as lat/lng — they're completely different number spaces
Validation
func isValidCoordinate(_ coord: CLLocationCoordinate2D) -> Bool {
coord.latitude >= -90 && coord.latitude <= 90
&& coord.longitude >= -180 && coord.longitude <= 180
&& !coord.latitude.isNaN && !coord.longitude.isNaN
}
If latitude > 90 or longitude > 180, coordinates are likely swapped or in wrong format.
Console Debugging
MapKit Logs
# View MapKit-related logs
log stream --predicate 'subsystem == "com.apple.MapKit"' --level debug
# Filter for your app
log stream --predicate 'process == "YourApp" AND (subsystem == "com.apple.MapKit" OR subsystem == "com.apple.CoreLocation")'
Common Console Messages
| Message | Meaning |
|---|---|
No renderer for overlay |
Missing rendererFor delegate method |
Reuse identifier not registered |
Call register before dequeue |
CLLocationManager authorizationStatus is denied |
User denied location |
Resources
WWDC: 2023-10043, 2024-10094
Docs: /mapkit, /mapkit/mklocalsearch
Skills: axiom-mapkit, axiom-mapkit-ref, axiom-core-location-diag
Weekly Installs
39
Repository
charleswiltgen/axiomGitHub Stars
636
First Seen
Feb 27, 2026
Security Audits
Installed on
opencode37
github-copilot37
codex37
amp37
cline37
kimi-cli37