एक ही बार में कई एनोटेशन दिखाने के लिए MKMapView पोजिशनिंग


93

मुझे कई एनोटेशन मिले हैं जिन्हें मैं अपने MKMapView में जोड़ना चाहता हूं (यह 0-n आइटम हो सकता है, जहां n आमतौर पर लगभग 5 है)। मैं एनोटेशन को ठीक से जोड़ सकता हूं, लेकिन मैं एक बार में सभी एनोटेशन को ऑनस्क्रीन फिट करने के लिए मानचित्र का आकार बदलना चाहता हूं, और मुझे यकीन नहीं है कि यह कैसे करना है।

मैं देख -regionThatFits:रहा हूं लेकिन मुझे यकीन नहीं है कि इसके साथ क्या करना है। अब तक जो भी मिला है, उसे दिखाने के लिए मैं कुछ कोड पोस्ट करूँगा। मुझे लगता है कि यह आम तौर पर सीधा काम होना चाहिए, लेकिन मैं अब तक MapKit के साथ थोड़ा परेशान महसूस कर रहा हूं।

- (void)locationManager:(CLLocationManager *)manager didUpdateToLocation:(CLLocation *)newLocation fromLocation:(CLLocation *)oldLocation{

location = newLocation.coordinate;
//One location is obtained.. just zoom to that location

MKCoordinateRegion region;
region.center = location;

//Set Zoom level using Span
MKCoordinateSpan span;
span.latitudeDelta = 0.015;
span.longitudeDelta = 0.015;
region.span = span;
// Set the region here... but I want this to be a dynamic size
// Obviously this should be set after I've added my annotations
[mapView setRegion:region animated:YES];

// Test data, using these as annotations for now
NSArray *arr = [NSArray arrayWithObjects:@"one", @"two", @"three", @"four", nil];
float ex = 0.01;
for (NSString *s in arr) {
    JBAnnotation *placemark = [[JBAnnotation alloc] initWithLat:(location.latitude + ex) lon:location.longitude];
    [mapView addAnnotation:placemark];
    ex = ex + 0.005;
}
    // What do I do here?
    [mapView setRegion:[mapView regionThatFits:region] animated:YES];
}

ध्यान दें, यह सब तब होता है जब मुझे एक स्थान अपडेट प्राप्त होता है ... मुझे नहीं पता कि यह करने के लिए उपयुक्त स्थान है या नहीं। यदि नहीं, तो बेहतर जगह कहां होगी? -viewDidLoad?

अग्रिम में धन्यवाद।

जवाबों:



137

लिंक जिम द्वारा पोस्ट की गई अब मर चुका है, लेकिन मैं कोड की खोज करने में सक्षम था (जो मैं बुकमार्क कहीं था)। उम्मीद है की यह मदद करेगा।

- (void)zoomToFitMapAnnotations:(MKMapView *)mapView { 
    if ([mapView.annotations count] == 0) return; 

    CLLocationCoordinate2D topLeftCoord; 
    topLeftCoord.latitude = -90; 
    topLeftCoord.longitude = 180; 

    CLLocationCoordinate2D bottomRightCoord; 
    bottomRightCoord.latitude = 90; 
    bottomRightCoord.longitude = -180; 

    for(id<MKAnnotation> annotation in mapView.annotations) { 
        topLeftCoord.longitude = fmin(topLeftCoord.longitude, annotation.coordinate.longitude); 
        topLeftCoord.latitude = fmax(topLeftCoord.latitude, annotation.coordinate.latitude); 
        bottomRightCoord.longitude = fmax(bottomRightCoord.longitude, annotation.coordinate.longitude); 
        bottomRightCoord.latitude = fmin(bottomRightCoord.latitude, annotation.coordinate.latitude); 
    } 

    MKCoordinateRegion region; 
    region.center.latitude = topLeftCoord.latitude - (topLeftCoord.latitude - bottomRightCoord.latitude) * 0.5; 
    region.center.longitude = topLeftCoord.longitude + (bottomRightCoord.longitude - topLeftCoord.longitude) * 0.5;      

    // Add a little extra space on the sides
    region.span.latitudeDelta = fabs(topLeftCoord.latitude - bottomRightCoord.latitude) * 1.1;
    region.span.longitudeDelta = fabs(bottomRightCoord.longitude - topLeftCoord.longitude) * 1.1; 

    region = [mapView regionThatFits:region]; 
    [mapView setRegion:region animated:YES]; 
}

14
मैं तुम्हें चूम सकता है। इसने मुझे समय का एक समूह बचा लिया। मैंने 1 स्थान को संभालने के लिए ऊपर दिए गए कोड को जोड़ा। यह थोड़ा करीब और व्यक्तिगत हो गया। मैं टिप्पणी के बाद से कोड को चबाने की प्रवृत्ति के रूप में पोस्ट करूंगा।
माइकल रीड

आपका बहुत बहुत धन्यवाद। मैंने इसे एक उपवर्ग में जोड़ा MKMapViewऔर विधि को बदल दिया - (void) zoomToFitAnnotations:(BOOL)animated। अच्छी तरह से काम!
simonbs

1
यह बहुत अच्छा काम कर रहा है। यह भी उपयोगी है। आप मूल्य में ज़ूम आउट या ज़ूम बदल सकते हैं। so Region.span.latitudeDelta = fabs (topLeftCoord.latitude - bottomRightCoord.latitude) * 1.1; /// बदलते सिद्धांत बदलते मूल्य, बदलती कीमत। जब आप मूल्य बढ़ाते हैं: ज़ूम आउट ........ जब आप मूल्य घटाते हैं: उदाहरण के लिए ज़ूम करें: Region.span.latitudeDelta = fabs (topLeftCoord.latitude - bottomRightCoord.latitude) 4.1;
एरहान डेमिरसी

1
@ MR.Mustafa: इसका काम, कमाल! लेकिन मुझे लगता है कि इस मुद्दे को हल करना काफी है। तो कृपया कोई मुझे समझाए कि यह कैसे काम करता है। या किसी भी लिंक के माध्यम से। क्षमा करें यदि मैं मूर्ख हूं, तो मैं शुरुआती हूं। Pls समर्थन करते हैं। धन्यवाद
सिद्धार्थ ओले

1
@ मुस्तफा ... धन्यवाद यह मेरा दिन बच गया।
Vvk

133

इतना जटिल क्यों?

MKCoordinateRegion coordinateRegionForCoordinates(CLLocationCoordinate2D *coords, NSUInteger coordCount) {
    MKMapRect r = MKMapRectNull;
    for (NSUInteger i=0; i < coordCount; ++i) {
        MKMapPoint p = MKMapPointForCoordinate(coords[i]);
        r = MKMapRectUnion(r, MKMapRectMake(p.x, p.y, 0, 0));
    }
    return MKCoordinateRegionForMapRect(r);
}

6
अविश्वसनीय कितना सरल, क्लीनर और आसान है यह पोस्ट किए गए विकल्पों की तुलना में है। वास्तव में, आप इसे और भी सरल बना सकते हैं क्योंकि MKCoordinateRegion में कनवर्ट करने की कोई आवश्यकता नहीं है - बस सेट करेंविज़नटैम्परेक्ट को कॉल करें: अपने MKMapView पर आपके द्वारा यहां बनाए गए MKMapRect के साथ।
लेंसोवेट

2
एनोटेशन कभी-कभी नक्शे के शीर्ष पर अटक जाते हैं और दिखाई नहीं देते हैं। MKCoordinateRegion बनने के बाद ज़ूम बढ़ाने के लिए सबसे अच्छा तरीका क्या है?
काइल सी

3
@KyleC[mapView setVisibleMapRect:mapRect edgePadding:UIEdgeInsetsMake(20.0f, 20.0f, 20.0f, 20.0f) animated:animated];
उपयोगकर्ता

आप CLLocationCoordinate2D *coordsसरणी कैसे बनाते हैं ? का उपयोग कर malloc()?
हुलंग

3
@KyleC। मैंने इसे वापस लौटने से पहले जोड़ा, rजो मूल रूप से 20 प्रतिशतCGFloat zoomOutPercent = 0.2f; r = MKMapRectMake(r.origin.x-r.size.width*zoomOutPercent, r.origin.y-r.size.height*zoomOutPercent, r.size.width*(1+zoomOutPercent*2), r.size.height*(1+zoomOutPercent*2));
लूज

44

मैंने इस क्षेत्र को ज़ूम आउट करने के लिए (या) एक क्षेत्र एनोटेशन और वर्तमान स्थान को शामिल करने के लिए कुछ किया है। आप अपने एनोटेशन के माध्यम से लूपिंग द्वारा इसका विस्तार कर सकते हैं।

बुनियादी कदम हैं:

  • न्यूनतम lat / long की गणना करें
  • अधिकतम lat / long की गणना करें
  • इन दो बिंदुओं के लिए CLLocation ऑब्जेक्ट बनाएँ
  • अंकों के बीच की दूरी की गणना करें
  • केंद्र बिंदु का उपयोग करके क्षेत्र बनाएँ और डिग्री में परिवर्तित दूरी
  • समायोजित करने के लिए MapView में क्षेत्र पास करें
  • MapView क्षेत्र सेट करने के लिए समायोजित क्षेत्र का उपयोग करें
    -(IBAction)zoomOut:(id)sender {

        CLLocationCoordinate2D southWest = _newLocation.coordinate;
        CLLocationCoordinate2D northEast = southWest;

        southWest.latitude = MIN(southWest.latitude, _annotation.coordinate.latitude);
        southWest.longitude = MIN(southWest.longitude, _annotation.coordinate.longitude);

        northEast.latitude = MAX(northEast.latitude, _annotation.coordinate.latitude);
        northEast.longitude = MAX(northEast.longitude, _annotation.coordinate.longitude);

        CLLocation *locSouthWest = [[CLLocation alloc] initWithLatitude:southWest.latitude longitude:southWest.longitude];
        CLLocation *locNorthEast = [[CLLocation alloc] initWithLatitude:northEast.latitude longitude:northEast.longitude];

        // This is a diag distance (if you wanted tighter you could do NE-NW or NE-SE)
        CLLocationDistance meters = [locSouthWest getDistanceFrom:locNorthEast];

        MKCoordinateRegion region;
        region.center.latitude = (southWest.latitude + northEast.latitude) / 2.0;
        region.center.longitude = (southWest.longitude + northEast.longitude) / 2.0;
        region.span.latitudeDelta = meters / 111319.5;
        region.span.longitudeDelta = 0.0;

        _savedRegion = [_mapView regionThatFits:region];
        [_mapView setRegion:_savedRegion animated:YES];

        [locSouthWest release];
        [locNorthEast release];
    }

यह जाने का रास्ता दिखता है। धन्यवाद!
jbrennan

1
का उपयोग कर काम करने के लिए यह प्रबंधित MKCoordinateRegionMake: gist.github.com/1599700 अगर कोई अभी भी इस तरह से करना चाहता है।
चक्र

Region.center.latitude = (southWest.latitude + NorthEast.latitude) / 2.0; इसके लिए धन्यवाद
टोनी

क्या यह मेरिडियन के दोनों ओर बिंदुओं के साथ काम करता है? भूमध्यरेखा?
एलियट

1
यह कोड उन स्थानों को ऑफ़स्क्रीन रखता है जब स्थानों का समान y- मान होता है। उदाहरण, (50, -4) और (100, -3) पर दो स्थानों को दिखाते हुए, नक्शे को बहुत दूर तक ज़ूम किया जाएगा, जिससे स्क्रीन के बाईं और दाईं ओर के निर्देशांक बंद हो जाएंगे।
यूजर

21

मेरे पास एक अलग उत्तर है। मैं खुद जूम-टू-फिट एल्गोरिदम को लागू करने जा रहा था, लेकिन मुझे लगा कि Apple के पास ऐसा करने का एक तरीका होना चाहिए जो हम इतने काम के बिना चाहते थे। एपीआई डोको का उपयोग करते हुए जल्दी से पता चला कि मैं एमकेपॉलिगो का उपयोग कर सकता था जो कि आवश्यक था:

/* this simply adds a single pin and zooms in on it nicely */
- (void) zoomToAnnotation:(MapAnnotation*)annotation {
    MKCoordinateSpan span = {0.027, 0.027};
    MKCoordinateRegion region = {[annotation coordinate], span};
    [mapView setRegion:region animated:YES];
}

/* This returns a rectangle bounding all of the pins within the supplied
   array */
- (MKMapRect) getMapRectUsingAnnotations:(NSArray*)theAnnotations {
    MKMapPoint points[[theAnnotations count]];

    for (int i = 0; i < [theAnnotations count]; i++) {
        MapAnnotation *annotation = [theAnnotations objectAtIndex:i];
        points[i] = MKMapPointForCoordinate(annotation.coordinate);
    }

    MKPolygon *poly = [MKPolygon polygonWithPoints:points count:[theAnnotations count]];

    return [poly boundingMapRect];
}

/* this adds the provided annotation to the mapview object, zooming 
   as appropriate */
- (void) addMapAnnotationToMapView:(MapAnnotation*)annotation {
    if ([annotations count] == 1) {
        // If there is only one annotation then zoom into it.
        [self zoomToAnnotation:annotation];
    } else {
        // If there are several, then the default behaviour is to show all of them
        //
        MKCoordinateRegion region = MKCoordinateRegionForMapRect([self getMapRectUsingAnnotations:annotations]);

        if (region.span.latitudeDelta < 0.027) {
            region.span.latitudeDelta = 0.027;
        }

        if (region.span.longitudeDelta < 0.027) {
            region.span.longitudeDelta = 0.027;
        }
        [mapView setRegion:region];
    }

    [mapView addAnnotation:annotation];
    [mapView selectAnnotation:annotation animated:YES];
}

उम्मीद है की यह मदद करेगा।


कोई समस्या नहीं। आम तौर पर एक बेहतर तरीका है यदि आप तैयार हैं और उस पर खर्च करने का समय है।
PKCLsoft

मैंने पाया कि यह पिंस को स्क्रीन के किनारे से थोड़ा बहुत दूर लगाता है। AnnotationsRegion.span.latitudeDelta = annotationsRegion.span.latitudeDelta * kEventMapDetailBorderFactor जोड़ने का प्रयास करें; सेटरीजन से ठीक पहले।
एडम एबरबैच

आप सही हैं @AdamEberbach, लेकिन ऐसा लगता है कि आपकी क्लिप में एक स्थिरांक शामिल है जो उपलब्ध नहीं है। क्या आपको एक मूल्य मिला जो पिंस के चारों ओर "अच्छा" बॉर्डर प्रदान करता है?
PKCLsoft

IOS7 के साथ नए शोऑनोटेशन विधि का उपयोग करने के बारे में नीचे दिए गए कोड कमांडर का जवाब एक अच्छा मार्जिन जोड़ता है, जो वास्तव में बेहतर काम करता है, हालांकि यह कोड कूलर है।
जेम्स टॉमी जूल

14

आप इसे इस तरह से भी कर सकते हैं ।।

// Position the map so that all overlays and annotations are visible on screen.
MKMapRect regionToDisplay = [self mapRectForAnnotations:annotationsToDisplay];
if (!MKMapRectIsNull(regionToDisplay)) myMapView.visibleMapRect = regionToDisplay;

- (MKMapRect) mapRectForAnnotations:(NSArray*)annotationsArray
{
    MKMapRect mapRect = MKMapRectNull;

    //annotations is an array with all the annotations I want to display on the map
    for (id<MKAnnotation> annotation in annotations) { 

        MKMapPoint annotationPoint = MKMapPointForCoordinate(annotation.coordinate);
        MKMapRect pointRect = MKMapRectMake(annotationPoint.x, annotationPoint.y, 0, 0);

        if (MKMapRectIsNull(mapRect)) 
        {
            mapRect = pointRect;
        } else 
        {
            mapRect = MKMapRectUnion(mapRect, pointRect);
        }
    }

     return mapRect;
}

13

सभी की जानकारी और सुझावों के आधार पर मैं निम्नलिखित के साथ आया। योगदान देने के लिए इस चर्चा में सभी के लिए धन्यवाद :) यह उस दृश्य नियंत्रक में जाएगा जिसमें mapView शामिल है।

- (void)zoomToFitMapAnnotations { 

if ([self.mapView.annotations count] == 0) return; 

int i = 0;
MKMapPoint points[[self.mapView.annotations count]];

//build array of annotation points
for (id<MKAnnotation> annotation in [self.mapView annotations])
        points[i++] = MKMapPointForCoordinate(annotation.coordinate);

MKPolygon *poly = [MKPolygon polygonWithPoints:points count:i];

[self.mapView setRegion:MKCoordinateRegionForMapRect([poly boundingMapRect]) animated:YES]; 
}

इसे ज्यादा वोट मिलने चाहिए। बहुत सटीक और बात तक।
नताशा

5

मेरे मामले में, मैं CLLocation ऑब्जेक्ट्स के साथ शुरू कर रहा हूं और उनमें से प्रत्येक के लिए एनोटेशन बना रहा हूं।
मुझे केवल दो एनोटेशन लगाने की आवश्यकता है, इसलिए मेरे पास पॉइंट्स की सरणी बनाने के लिए एक सरल तरीका है, लेकिन यह आसानी से एक सरणी बनाने के लिए विस्तारित किया जा सकता है, जिसमें एक मनमाना लंबाई के साथ CLLocations का एक सेट दिया गया है।

यहां मेरा कार्यान्वयन है (MKMapPoints बनाने की आवश्यकता नहीं है):

//start with a couple of locations
CLLocation *storeLocation = store.address.location.clLocation;
CLLocation *userLocation = [LBLocationController sharedController].currentLocation;

//build an array of points however you want
CLLocationCoordinate2D points[2] = {storeLocation.coordinate, userLocation.coordinate};

//the magic part
MKPolygon *poly = [MKPolygon polygonWithCoordinates:points count:2];
[self.mapView setRegion:MKCoordinateRegionForMapRect([poly boundingMapRect])];

5

स्विफ्ट, एक बहुभुज, और कुछ अतिरिक्त पैडिंग का उपयोग करके मैंने निम्नलिखित का उपयोग किया:

func zoomToFit() {
    var allLocations:[CLLocationCoordinate2D] = [
        CLLocationCoordinate2D(latitude: 32.768805, longitude: -117.167119),
        CLLocationCoordinate2D(latitude: 32.770480, longitude: -117.148385),
        CLLocationCoordinate2D(latitude: 32.869675, longitude: -117.212929)
    ]

    var poly:MKPolygon = MKPolygon(coordinates: &allLocations, count: allLocations.count)

    self.mapView.setVisibleMapRect(poly.boundingMapRect, edgePadding: UIEdgeInsetsMake(40.0, 40.0, 40.0, 40.0), animated: false)
}


setVisibleMapRect (...)। मैं खुद गणित कर रहा था ... बेचारा।
कोडरॉवल

3

'MKMapView' में iOS 7 के रूप में एक नई विधि है जिसका आप उपयोग कर सकते हैं

घोषणा

स्विफ्ट

func showAnnotations(_ annotations: [AnyObject]!,
            animated animated: Bool)

उद्देश्य सी

- (void)showAnnotations:(NSArray *)annotations
               animated:(BOOL)animated

पैरामीटर

एनोटेशन जिन एनोटेशन को आप मानचित्र में देखना चाहते हैं। एनिमेटेड यस यदि आप चाहते हैं कि मानचित्र क्षेत्र एनिमेटेड रूप से बदल जाए, या नहीं, तो यदि आप चाहते हैं कि मानचित्र बिना एनिमेशन के तुरंत नया क्षेत्र प्रदर्शित करे।

विचार-विमर्श

इस पद्धति को कॉल करने से क्षेत्र की संपत्ति में मूल्य और नए मानचित्र क्षेत्र को प्रतिबिंबित करने के लिए संभावित रूप से अन्य गुण अपडेट होते हैं।


3

मुझे पता है कि यह एक पुराना सवाल है लेकिन, यदि आप मानचित्र का उपयोग करने वाले सभी एनोटेशन को प्रदर्शित करना चाहते हैं:

 mapView.showAnnotations(mapView.annotations, animated: true)

3

यहां मुस्तफा के जवाब के लिए स्विफ्ट इक्विवेलेंट (पुष्टि कार्य में: Xcode6.1, SDK 8.2) है:

func zoomToFitMapAnnotations() {
    if self.annotations.count == 0 {return}

    var topLeftCoordinate = CLLocationCoordinate2D(latitude: -90, longitude: 180)
    var bottomRightCoordinate = CLLocationCoordinate2D(latitude: 90, longitude: -180)

    for object in self.annotations {
        if let annotation = object as? MKAnnotation {
            topLeftCoordinate.longitude = fmin(topLeftCoordinate.longitude, annotation.coordinate.longitude)
            topLeftCoordinate.latitude = fmax(topLeftCoordinate.latitude, annotation.coordinate.latitude)
            bottomRightCoordinate.longitude = fmax(bottomRightCoordinate.longitude, annotation.coordinate.longitude)
            bottomRightCoordinate.latitude = fmin(bottomRightCoordinate.latitude, annotation.coordinate.latitude)
        }
    }

    let center = CLLocationCoordinate2D(latitude: topLeftCoordinate.latitude - (topLeftCoordinate.latitude - bottomRightCoordinate.latitude) * 0.5, longitude: topLeftCoordinate.longitude - (topLeftCoordinate.longitude - bottomRightCoordinate.longitude) * 0.5)

    print("\ncenter:\(center.latitude) \(center.longitude)")
    // Add a little extra space on the sides
    let span = MKCoordinateSpanMake(fabs(topLeftCoordinate.latitude - bottomRightCoordinate.latitude) * 1.01, fabs(bottomRightCoordinate.longitude - topLeftCoordinate.longitude) * 1.01)
    print("\nspan:\(span.latitudeDelta) \(span.longitudeDelta)")

    var region = MKCoordinateRegion(center: center, span: span)


    region = self.regionThatFits(region)

    self.setRegion(region, animated: true)

}

1
हे iOS_Developer। स्विफ्ट रूपांतरण के लिए धन्यवाद। मेरे लिए यह काम नहीं कर रहा है क्योंकि मुझे लगता है कि आपको topLeftCoordinate.latitude और bottomRightCoordinate.longitude के लिए "fmin" के बजाय दो "fmax" याद आ रहे हैं।
फिलिप ओटो

2

एक संभावित समाधान वर्तमान स्थान और सभी एनोटेशन के बीच की दूरी को मापने और MKCoordinateRegionMakeWithDistance विधि का उपयोग करके एक क्षेत्र बनाने के लिए हो सकता है जिसमें सबसे अधिक दूरी एनोटेशन की तुलना में थोड़ी अधिक दूरी हो।

हालांकि यह आपके द्वारा जोड़े गए अधिक एनोटेशन को धीमा कर देगा।


मैं केवल स्वयं को मान्य करने के लिए टिप्पणी अनुभाग से गुजर रहा था। खुशी है कि किसी और को लगता है कि मैंने जिस तरह से किया था :-) चूंकि मैं केवल दो एनोटेशन (शुरुआत और अंत बिंदु) जोड़ता हूं, मुझे कोई सुस्ती महसूस नहीं हुई।
थंडासोरू

2
- (void)zoomToFitMapAnnotations {

if ([self.mapview.annotations count] == 0) return;

int i = 0;
MKMapPoint points[[self.mapview.annotations count]];

//build array of annotation points
for (id<MKAnnotation> annotation in [self.mapview annotations])
    points[i++] = MKMapPointForCoordinate(annotation.coordinate);

MKPolygon *poly = [MKPolygon polygonWithPoints:points count:i];

[self.mapview setRegion:MKCoordinateRegionForMapRect([poly boundingMapRect]) animated:YES];
}

2

me2(स्विफ्ट में अब) के उत्कृष्ट उत्तर के आधार पर

func coordinateRegionForCoordinates(coords: [CLLocationCoordinate2D]) -> MKCoordinateRegion {
    var rect: MKMapRect = MKMapRectNull
    for coord in coords {
        let point: MKMapPoint = MKMapPointForCoordinate(coord)
        rect = MKMapRectUnion(rect, MKMapRectMake(point.x, point.y, 0, 0))
    }
    return MKCoordinateRegionForMapRect(rect)
}

1

1 स्थान को संभालने के लिए क्लॉज़ को थोड़ा जोड़ा गया- मुस्तफ़ा के कोड कोड स्निपेट में जोड़ने के लिए। उसके लिए pkclSoft के zoomToAnnotation फ़ंक्शन का उपयोग किया:

if ([mapView.annotations count] == 1){
    MKCoordinateSpan span = {0.027, 0.027};
    region.span = span;
    CLLocationCoordinate2D singleCoordinate = [[mapView.annotations objectAtIndex:0] coordinate];
    region.center.latitude = singleCoordinate.latitude;
    region.center.longitude = singleCoordinate.longitude;
}
else
{
    // mustufa's code
}

1

यह कोड मेरे लिए काम करता है, यह वर्तमान स्थान के साथ सभी पिन दिखाता है, आशा है कि यह आपकी मदद करता है,

func setCenterForMap() {
    var mapRect: MKMapRect = MKMapRectNull
    for loc in mapView.annotations {
        let point: MKMapPoint = MKMapPointForCoordinate(loc.coordinate)
        print( "location is : \(loc.coordinate)");
        mapRect = MKMapRectUnion(mapRect, MKMapRectMake(point.x,point.y,0,0))
    }
    if (locationManager.location != nil) {
        let point: MKMapPoint = MKMapPointForCoordinate(locationManager.location!.coordinate)
        print( "Cur location is : \(locationManager.location!.coordinate)");
        mapRect = MKMapRectUnion(mapRect, MKMapRectMake(point.x,point.y,0,0))
    }

    mapView.setVisibleMapRect(mapRect, edgePadding: UIEdgeInsetsMake(40.0, 40.0, 40.0, 40.0), animated: true)

}

0

मुझे उम्मीद है कि यह कम से कम प्रासंगिक है, यह वही है जो मैंने मोनो के लिए एक साथ रखा (pkclSoft के उत्तर के आधार पर):

void ZoomMap (MKMapView map)
{
    var annotations = map.Annotations;

    if (annotations == null || annotations.Length == 0) 
        return;

    var points = annotations.OfType<MapAnnotation> ()
                            .Select (s => MKMapPoint.FromCoordinate (s.Coordinate))
                            .ToArray ();            

    map.SetVisibleMapRect(MKPolygon.FromPoints (points).BoundingMapRect, true); 
}

0
CLLocationCoordinate2D min = CLLocationCoordinate2DMake(99999.0, 99999.0);
CLLocationCoordinate2D max = CLLocationCoordinate2DMake(-99999.0, -99999.0);

// find max/min....

// zoom to cover area
// TODO: Maybe better using a MKPolygon which can calculate its own fitting region.
CLLocationCoordinate2D center = CLLocationCoordinate2DMake((max.latitude + min.latitude) / 2.0, (max.longitude + min.longitude) / 2.0);
MKCoordinateSpan span = MKCoordinateSpanMake(max.latitude - min.latitude, max.longitude - min.longitude);
MKCoordinateRegion region = MKCoordinateRegionMake(center, span);

[_mapView setRegion:[_mapView regionThatFits:region] animated:YES];

0

Me2 प्रतिक्रिया के आधार पर मैंने MKMapView के लिए कुछ मार्जिन जोड़ने और उपयोगकर्ता स्थान एनोटेशन को छोड़ने के लिए एक श्रेणी लिखी है:

@interface MKMapView (ZoomToFitAnnotations)
- (void)zoomToFitAnnotations:(BOOL)animated;
@end

@implementation MKMapView (ZoomToFitAnnotations)
- (void)zoomToFitAnnotations:(BOOL)animated {
    if (self.annotations.count == 0)
        return;

    MKMapRect rect = MKMapRectNull;
    for (id<MKAnnotation> annotation in self.annotations) {
        if ([annotation isKindOfClass:[MKUserLocation class]] == false) {
            MKMapPoint point = MKMapPointForCoordinate(annotation.coordinate);
            rect = MKMapRectUnion(rect, MKMapRectMake(point.x, point.y, 0, 0));
        }
    }

    MKCoordinateRegion region = MKCoordinateRegionForMapRect(rect);
    region.span.longitudeDelta *= 2; // Margin
    region.span.latitudeDelta *= 2; // Margin
    [self setRegion:region animated:animated];
}
@end

0

चूँकि मैं किसी उत्तर पर टिप्पणी नहीं कर सकता, इसलिए मैं अपनी सुविधा को @ me2 के उत्तर में जोड़ना चाहूंगा (क्योंकि मुझे लगा कि यह सबसे सुंदर दृष्टिकोण है जो यहां मिला)।

मेरी व्यक्तिगत परियोजना के लिए, मैंने केवल एमकेपॉपर व्यू वर्ग पर एक वर्जन कॉमन ऑपरेशन के लिए "दृश्यमान क्षेत्र" कार्यक्षमता को एनकैप्सुलेट करने के लिए एक श्रेणी जोड़ी: एमकेपॉपर व्यू उदाहरण पर सभी वर्तमान में लोड किए गए एनोटेशन को देखने में सक्षम होने के लिए सेटिंग। परिणाम यह था:

.h फाइल

#import <MapKit/MapKit.h>

@interface MKMapView (Extensions)

-(void)ij_setVisibleRectToFitAllLoadedAnnotationsAnimated:(BOOL)animated;
-(void)ij_setVisibleRectToFitAnnotations:(NSArray *)annotations animated:(BOOL)animated;


@end

.m फ़ाइल

#import "MKMapView+Extensions.h"

@implementation MKMapView (Extensions)

/**
 *  Changes the currently visible portion of the map to a region that best fits all the currently loadded annotations on the map, and it optionally animates the change.
 *
 *  @param animated is the change should be perfomed with an animation.
 */
-(void)ij_setVisibleRectToFitAllLoadedAnnotationsAnimated:(BOOL)animated
{
    MKMapView * mapView = self;

    NSArray * annotations = mapView.annotations;

    [self ij_setVisibleRectToFitAnnotations:annotations animated:animated];

}


/**
 *  Changes the currently visible portion of the map to a region that best fits the provided annotations array, and it optionally animates the change.
    All elements from the array must conform to the <MKAnnotation> protocol in order to fetch the coordinates to compute the visible region of the map.
 *
 *  @param annotations an array of elements conforming to the <MKAnnotation> protocol, holding the locations for which the visible portion of the map will be set.
 *  @param animated    wether or not the change should be perfomed with an animation.
 */
-(void)ij_setVisibleRectToFitAnnotations:(NSArray *)annotations animated:(BOOL)animated
{
    MKMapView * mapView = self;

    MKMapRect r = MKMapRectNull;
    for (id<MKAnnotation> a in annotations) {
        ZAssert([a conformsToProtocol:@protocol(MKAnnotation)], @"ERROR: All elements of the array MUST conform to the MKAnnotation protocol. Element (%@) did not fulfill this requirement", a);
        MKMapPoint p = MKMapPointForCoordinate(a.coordinate);
        //MKMapRectUnion performs the union between 2 rects, returning a bigger rect containing both (or just one if the other is null). here we do it for rects without a size (points)
        r = MKMapRectUnion(r, MKMapRectMake(p.x, p.y, 0, 0));
    }

    [mapView setVisibleMapRect:r animated:animated];

}

@end

जैसा कि आप देख सकते हैं, मैंने अब तक 2 विधियाँ जोड़ी हैं: एक मानचित्र के दृश्य क्षेत्र को उस एक पर सेट करने के लिए, जो MKMapView उदाहरण पर सभी वर्तमान में लोड किए गए एनोटेशन को फिट करता है, और इसे किसी भी अन्य ऑब्जेक्ट पर सेट करने के लिए एक और विधि है। तो mapView के दृश्य क्षेत्र को सेट करने के लिए कोड फिर उतना ही सरल होगा:

   //the mapView instance  
    [self.mapView ij_setVisibleRectToFitAllLoadedAnnotationsAnimated:animated]; 

मुझे आशा है कि यह मदद करता है =)


0

इस विस्तार पर विचार करें:

extension MKCoordinateRegion {
    init(locations: [CLLocationCoordinate2D], marginMultiplier: Double = 1.1) {
        let mapRect = locations.reduce(MKMapRect(), {
            let point = MKMapPointForCoordinate($1)
            let rect = MKMapRect(origin: point, size: MKMapSize(width: 0.0, height: 0.0))
            return MKMapRectUnion($0, rect)
        })

        var coordinateRegion = MKCoordinateRegionForMapRect(mapRect)
        coordinateRegion.span.latitudeDelta *= marginMultiplier
        coordinateRegion.span.longitudeDelta *= marginMultiplier
        self = coordinateRegion
    }
}

0

एक तेज 5 संस्करण:

   func regionFor(coordinates coords: [CLLocationCoordinate2D]) -> MKCoordinateRegion {
        var r = MKMapRect.null

        for i in 0 ..< coords.count {
            let p = MKMapPoint(coords[i])

            r = r.union(MKMapRect(x: p.x, y: p.y, width: 0, height: 0))
        }

        return MKCoordinateRegion(r)
    }

0

स्टीफन डी लुका के जवाब में अतिरिक्त जोड़ना । यहां हम MKMapView को फिट करने के लिए UserLocation प्लस कस्टम एनोटेशन रख सकते हैं

 private func centerViewOnUserLocation() {
    
    if selectedLatitudeFromPreviousVC?.description != nil && selectedLongitudeFromPreviousVC?.description != nil {
        
        if let location = locationManager.location?.coordinate {
            let region = regionFor(coordinates: [
                CLLocationCoordinate2D(latitude: selectedLatitudeFromPreviousVC!, longitude: selectedLongitudeFromPreviousVC!),
                location])
            mkmapView.setRegion(region, animated: true)
        }
    } else {
        if let location = locationManager.location?.coordinate {
            let region = MKCoordinateRegion.init(center: location, latitudinalMeters: regionInMeters, longitudinalMeters: regionInMeters)
            mkmapView.setRegion(region, animated: true)
        }
    }
}

private func regionFor(coordinates coords: [CLLocationCoordinate2D]) -> MKCoordinateRegion {
    var r = MKMapRect.null

    for i in 0 ..< coords.count {
        let p = MKMapPoint(coords[i])
        r = r.union(MKMapRect(x: p.x, y: p.y, width: 0, height: 0))
    }

    var coordinateRegion = MKCoordinateRegion(r)
    coordinateRegion.span.latitudeDelta *= 1.5
    coordinateRegion.span.longitudeDelta *= 1.5
    return coordinateRegion
}

यहां छवि विवरण दर्ज करें

हमारी साइट का प्रयोग करके, आप स्वीकार करते हैं कि आपने हमारी Cookie Policy और निजता नीति को पढ़ और समझा लिया है।
Licensed under cc by-sa 3.0 with attribution required.