This snapshot establishes the camera-to-result recognition flow and related tests while checking in the project skill/docs assets required for the configured local tooling.
463 lines
15 KiB
Markdown
463 lines
15 KiB
Markdown
---
|
|
name: axiom-mapkit-diag
|
|
description: MapKit troubleshooting — annotations not appearing, region jumping, clustering not working, search failures, overlay rendering issues, user location problems
|
|
license: MIT
|
|
metadata:
|
|
version: "1.0.0"
|
|
last-updated: "2026-02-26"
|
|
---
|
|
|
|
# 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-patterns
|
|
- `axiom-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.
|
|
|
|
```swift
|
|
// ❌ 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)` and `coordinate` property on MKMapPoint
|
|
- Never use MKMapPoint x/y as lat/lng — they're completely different number spaces
|
|
|
|
### Validation
|
|
|
|
```swift
|
|
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
|
|
|
|
```bash
|
|
# 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
|