【问题标题】:Searching Annotations in MapView在 MapView 中搜索注解
【发布时间】:2017-03-25 04:31:57
【问题描述】:

我关注了how-to-search-for-location-using-apples-mapkit,关于在mapView 中搜索annotations 并使用MKLocalSearch 进行全球搜索。

但是,我不想用 MKLocalSearch 搜索,而是搜索我自己的 annotations 我添加了自己,例如:

let LitzmanLocation = CLLocationCoordinate2DMake(32.100668,34.775192)
        // Drop a pin
        let Litzman = MKPointAnnotation()
        Litzman.coordinate = LitzmanLocation
        Litzman.title = "Litzman Bar"
        Litzman.subtitle = "נמל תל אביב 18,תל אביב"
        mapView.addAnnotation(Litzman)

        let ShalvataLocation = CLLocationCoordinate2DMake(32.101145,34.775163)
        // Drop a pin
        let Shalvata = MKPointAnnotation()
        Shalvata.coordinate = ShalvataLocation
        Shalvata.title = "Shalvata"
        Shalvata.subtitle = "האנגר 28,נמל תל אביב"
        mapView.addAnnotation(Shalvata)


        let MarkidLocation = CLLocationCoordinate2DMake(32.074961,34.781679)
        // Drop a pin
        let Markid = MKPointAnnotation()
        Markid.coordinate = MarkidLocation
        Markid.title = "Markid"
        Markid.subtitle = "אבן גבירול 30,תל אביב"
        mapView.addAnnotation(Markid)

这是我的代码:

MapViewController.Swift:

import UIKit
import MapKit
import CoreLocation

protocol HandleMapSearch {
    func dropPinZoomIn(placemark:MKPlacemark)
}

class MapViewController: UIViewController,MKMapViewDelegate, CLLocationManagerDelegate,UISearchBarDelegate{


    @IBOutlet var mapView: MKMapView!

    var resultSearchController:UISearchController? = nil

    var selectedPin:MKPlacemark? = nil

    @IBAction func MapSearchController(sender: AnyObject) {
        resultSearchController!.hidesNavigationBarDuringPresentation = false
        self.resultSearchController!.searchBar.delegate = self
        presentViewController(resultSearchController!, animated: true, completion: nil)
        self.resultSearchController!.searchBar.barTintColor = UIColor.blackColor()
        self.resultSearchController!.searchBar.placeholder = "חפש ברים"
        self.resultSearchController!.dimsBackgroundDuringPresentation = true
        self.resultSearchController!.searchBar.sizeToFit()
    }
override func viewDidLoad() {
        super.viewDidLoad()

        let locationSearchTable = storyboard!.instantiateViewControllerWithIdentifier("LocationSearchTable") as! LocationSearchTable
        resultSearchController = UISearchController(searchResultsController: locationSearchTable)
        resultSearchController?.searchResultsUpdater = locationSearchTable

        locationSearchTable.mapView = mapView

        locationSearchTable.handleMapSearchDelegate = self
}
}
}

extension MapViewController: HandleMapSearch {
    func dropPinZoomIn(placemark:MKPlacemark){
        // cache the pin
        selectedPin = placemark
        // clear existing pins
        mapView.removeAnnotations(mapView.annotations)
        let annotation = MKPointAnnotation()
        annotation.coordinate = placemark.coordinate
        annotation.title = placemark.name
        if let city = placemark.locality,
            let state = placemark.administrativeArea {
            annotation.subtitle = "(city) (state)"
        }
        mapView.addAnnotation(annotation)
        let span = MKCoordinateSpanMake(0.05, 0.05)
        let region = MKCoordinateRegionMake(placemark.coordinate, span)
        mapView.setRegion(region, animated: true)
    }
}

LocalSearchTable.Swift:

import UIKit
import MapKit

class LocationSearchTable : UITableViewController {

    var matchingItems:[MKMapItem] = []
    var mapView: MKMapView? = nil

    var handleMapSearchDelegate:HandleMapSearch? = nil

    func parseAddress(selectedItem:MKPlacemark) -> String {
        // put a space between "4" and "Melrose Place"
        let firstSpace = (selectedItem.subThoroughfare != nil && selectedItem.thoroughfare != nil) ? " " : ""
        // put a comma between street and city/state
        let comma = (selectedItem.subThoroughfare != nil || selectedItem.thoroughfare != nil) && (selectedItem.subAdministrativeArea != nil || selectedItem.administrativeArea != nil) ? ", " : ""
        // put a space between "Washington" and "DC"
        let secondSpace = (selectedItem.subAdministrativeArea != nil && selectedItem.administrativeArea != nil) ? " " : ""
        let addressLine = String(
            format:"%@%@%@%@%@%@%@",
            // street number
            selectedItem.subThoroughfare ?? "",
            firstSpace,
            // street name
            selectedItem.thoroughfare ?? "",
            comma,
            // city
            selectedItem.locality ?? "",
            secondSpace,
            // state
            selectedItem.administrativeArea ?? ""
        )
        return addressLine
    }

    }

extension LocationSearchTable : UISearchResultsUpdating {
    func updateSearchResultsForSearchController(searchController: UISearchController) {
        guard let mapView = mapView,
            let searchBarText = searchController.searchBar.text else { return }
        let request = MKLocalSearchRequest()
        request.naturalLanguageQuery = searchBarText
        request.region = mapView.region
        let search = MKLocalSearch(request: request)
        search.startWithCompletionHandler { response, _ in
            guard let response = response else {
                return
            }
            self.matchingItems = response.mapItems
            self.tableView.reloadData()
        }
    }

}

extension LocationSearchTable {
    override func tableView(tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
        return matchingItems.count
    }

    override func tableView(tableView: UITableView, cellForRowAtIndexPath indexPath: NSIndexPath) -> UITableViewCell {
        let cell = tableView.dequeueReusableCellWithIdentifier("MapSearchCell")!
        let selectedItem = matchingItems[indexPath.row].placemark
        cell.textLabel?.text = selectedItem.name
        cell.detailTextLabel?.text = parseAddress(selectedItem)
        return cell
    }
}

extension LocationSearchTable {
    override func tableView(tableView: UITableView, didSelectRowAtIndexPath indexPath: NSIndexPath) {
        let selectedItem = matchingItems[indexPath.row].placemark
        handleMapSearchDelegate?.dropPinZoomIn(selectedItem)
        dismissViewControllerAnimated(true, completion: nil)
    }
}

【问题讨论】:

  • 什么?你有一些数据源可以搜索吗? “搜索注释”是什么意思?
  • 我在我的 mapView 中有 100 个注释,并且逐个点击以找到我正在寻找的那个很烦人,所以我不知道如何真正但我想创建某种搜索。现在明白了吗?
  • 按标题过滤注释并创建搜索功能
  • 不,您的注释是如何进入地图的?您有一些数据源并使用注释填充地图?为什么您无法搜索该数据源并想搜索注释?
  • 我将能够在几个小时内发布我的代码

标签: ios iphone swift annotations mkmapview


【解决方案1】:

你只是参考这个链接 http://www.coderzheaven.com/2016/02/14/mapkit-demo-swift-annotation-custom-annotation-custom-annotation-with-button-search-showing-directions-apple-maps-ios/

您将了解以下主题

  • 在地图中显示注释。
  • 在地图中显示自定义注释。
  • 在地图中使用自定义按钮显示自定义注释。

【讨论】:

    【解决方案2】:

    TL;DR

    示例项目:https://github.com/JakubMazur/SO40539590


    好的,所以一开始我建议您将数据与控制器代码分开。我选择json 格式作为最通用的格式。所以:

    [
       {
          "title":"Litzman Bar",
          "subtitle":"נמל תל אביב 18,תל אביב",
          "coordinates":{
             "lat":32.100668,
             "lon":34.775192
          }
       },
       {
          "title":"Shalvata",
          "subtitle":"האנגר 28,נמל תל אביב",
          "coordinates":{
             "lat":32.101145,
             "lon":34.775163
          }
       },
       {
          "title":"Markid",
          "subtitle":"אבן גבירול 30,תל אביב",
          "coordinates":{
             "lat":32.074961,
             "lon":34.781679
          }
       }
    ]
    

    这基本上就是你的数据库。

    现在让我们将其解析为Array,以便在您的ViewConttroller 中使用。我再次建议您将其拆分为模型对象,例如 LocationCoordinate。让我们以其中的一类为例:

    class Location: NSObject {
    
        var title : String = String()
        var subtitle : String = String()
        var coordinates : Coordinate = Coordinate()
    
        public class func locationFromDictionary(_ dictionary : Dictionary<String, AnyObject>) -> Location {
            let location : Location = Location()
            location.title = dictionary["title"] as! String
            location.subtitle = dictionary["subtitle"] as! String
            location.coordinates = Coordinate.coordinateFromDictionary(dictionary["coordinates"] as! Dictionary<String, AnyObject>)
            return location;
        }
    }
    

    我不会将解析json 文件的代码粘贴到这个对象上,因为这不是这个问题的意义所在。您将在存储库中找到。

    现在让我们专注于问题。

    我会建议你不要搜索注解,而是搜索你的数据模型并在需要时重绘注解

    为了做到这一点(我将使用UISearchBar):

    func searchBarSearchButtonClicked(_ searchBar: UISearchBar) {
        self.displayArray = self.locationsDatabase.filter() {
            return $0.title.contains("i")
        }
    

    然后,当您像这样覆盖 setter 时:

    var displayArray : Array<Location> = [] {
        didSet {
            self.mapView .removeAnnotations(self.mapView.annotations)
            for location in displayArray {
                let coords : Coordinate = location.coordinates
                let point = MKPointAnnotation()
                point.coordinate = CLLocationCoordinate2DMake(CLLocationDegrees(coords.latitude),CLLocationDegrees(coords.longitude))
                point.title = location.title
                point.subtitle = location.subtitle
                mapView.addAnnotation(point)
            }
        }
    }
    

    您可以重新绘制注释。边缘情况,如空搜索字段、区分大小写、关闭键盘我留给你。但希望你能得到大致的想法。您可能认为它过度设计,但将对象作为单独的类和通用格式作为数据输入可能会在未来受益。

    完整项目:https://github.com/JakubMazur/SO40539590

    【讨论】:

    • 嗨,谢谢你的回答!正如我所说,我有一个更新的问题,我现在有什么代码,我刚刚编辑了这个问题并使它与新问题相同。其次,我需要 searchannotations 而不是绘制新的或使用MKLocalSearch,因为mapView 向用户显示附近餐厅的位置,所以我需要搜索annotations,当我选择结果时,它会zoomcenter 在正确的annotation 上.我希望你明白我的意思。
    • **按标题过滤注释
    • 是的,我仍然建议您将这家餐厅输入Array 并搜索此数组并在需要时更新地图。如果你坚持搜索显示的注释 mapView 有一个数组 annotations 显示所有注释。但是搜索它们有很多缺点,例如 - 如果用户搜索一次并想再次搜索会发生什么?您将隐藏注释并且无法在此处再次搜索。您必须从某处获取此数据以显示 - 搜索此堆栈而不是注释。
    • 如果我想再次搜索,我只需键入我想要的结果,它会缩放并以我想要的结果为中心 - 至少这就是我想要做的,如果我不这样做想再次搜索,我只需单击我拥有的另一个选项卡或缩小并手动选择
    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2011-06-03
    • 2018-11-20
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多