首页 > 解决方案 > 您如何更改 Google Maps 的集群图标?

问题描述

我有一个 GMSMapView 类型的 Google 地图,并且我在该地图上显示了标记,并且我还为该 mapView 实现了 Google 集群。

所以我真正想要实现的是我想点击特定集群的集群图标和图标更改。但是目前这并没有发生,因为当我点击选定的标记时,只有单个标记会发生变化,而不是集群 Item 会发生变化。

地图上显示的我的集群组

我的集群地图图片链接

我真正想要达到的目标

我想在集群点击图像链接上实现什么

我为个人标记实际取得的成就。

我为单个标记图像链接所取得的成就

所以上面的图像是集群标记我想在该集群项目上呈现一个图标但不幸的是它没有呈现为我无法在GMUClusterManagerDelegateGMUClusterRendererDelegate的代表中访问的图像。我不知道如何在集群标记的点击操作上更改集群图标。

注意:-我想更改集群项目的集群点击图标并且我已经为选定的单个标记实现了

下面是我到目前为止所做的代码

   override func viewDidLoad() {
        super.viewDidLoad()
        // Do any additional setup after loading the view.
        propertyMapView.delegate = self
        createListOfPOIItems()
    }

    // MARK : -- Map related functions
//creating POOitems for the map
    func createListOfPOIItems() {
        self.propertyPOIItemList = []
        for index in 0..<self.propertyListModalArray.count {
            let propertylistmodal = self.propertyListModalArray[index]
            if UtilitySharedClass.sharedInstance.isvalidGeoLat(propertylistmodal.propertylat) && UtilitySharedClass.sharedInstance.isvalidGeoLong(propertylistmodal.propertyLong) {
                let coordinate = CLLocationCoordinate2D(latitude: CLLocationDegrees(Float(propertylistmodal.propertylat!)!), longitude: CLLocationDegrees(Float(propertylistmodal.propertyLong!)!))
                if universityLocation == coordinate{
                    continue
                }
                let item = POIItem(position: coordinate, name: propertylistmodal.propertyName ?? "", propertyListModal: propertylistmodal)
                self.propertyPOIItemList?.append(item)
            }
        }
        initiateClustering()
    }

    func initiateClustering() {
        // Set up the cluster manager with the supplied icon generator and
        // renderer.
        let iconGenerator = ClusterIconGenerator()
        let algorithm = GMUNonHierarchicalDistanceBasedAlgorithm()
        let renderer = GMUDefaultClusterRenderer(mapView: propertyMapView,
                                                 clusterIconGenerator: iconGenerator)
        clusterManager = GMUClusterManager(map: propertyMapView, algorithm: algorithm,
                                           renderer: renderer)
        renderer.delegate = self
        renderer.minimumClusterSize = 2
        
        //renderer.animationDuration = 0.2
        
        // Generate and add random items to the cluster manager.
        generateClusterItems()
        
        // Call cluster() after items have been added to perform the clustering
        // and rendering on map.
        clusterManager.cluster()
        
        clusterManager.setDelegate(self, mapDelegate: self)
        
        self.setUniversityLocation()
    }

为地图添加集群

    /// adds property location to the cluster manager.
    private func generateClusterItems() {
        guard let items = self.propertyPOIItemList else { return }
        items.forEach({ (item) in
            clusterManager?.add(item)
            clusterManager?.cluster()
        })
    }

为集群实现委托

extension SearchViewController:  GMSMapViewDelegate, GMUClusterManagerDelegate, GMUClusterRendererDelegate {
    
    func renderer(_ renderer: GMUClusterRenderer, markerFor object: Any) -> GMSMarker? {
        let marker = CustomMarker()
        if let itemMarker = object as? GMUCluster {
            marker.markerType = .clusterType
            marker.markerCount = itemMarker.items.count
        } else if let itemMarker = object as? POIItem {
            marker.markerType = .itemType
            itemMarker.marker = marker
            marker.propertyListModal = itemMarker.propertyListModal
        }
        return marker
    }
    
    func renderer(_ renderer: GMUClusterRenderer, didRenderMarker marker: GMSMarker) {
        if let customMarker = marker as? CustomMarker {
            if customMarker.markerType == .itemType {
                let propertyMarker = createMarkerView(isSelected: customMarker.isSelected)
                propertyMarker.markerPriceLabel.text = customMarker.propertyListModal?.propertyPrice
                marker.icon = propertyMarker.asImage()
                marker.groundAnchor = CGPoint(x: 0.5, y: 1)
            }
        }
    }
    
    // MARK: - GMUClusterManagerDelegate
    func clusterManager(_ clusterManager: GMUClusterManager, didTap cluster: GMUCluster) -> Bool {
        let newCamera = GMSCameraPosition.camera(withTarget: cluster.position,
                                                 zoom: propertyMapView.camera.zoom)
        let update = GMSCameraUpdate.setCamera(newCamera)
        propertyMapView.moveCamera(update)

        isClusterTapped = true

        clusterPropertyList = []
        let properties = cluster.items
        properties.forEach({ (property) in
            if let propertyModal = property as? POIItem, propertyModal.propertyListModal != nil{
                clusterPropertyList.append(propertyModal.propertyListModal!)
            }
        })
        
        if let previousMarker = propertyMapView.selectedMarker as? CustomMarker{
            previousMarker.isSelected = !previousMarker.isSelected
            let propertyMarker = createMarkerView(isSelected: previousMarker.isSelected)
            propertyMarker.markerPriceLabel.text = previousMarker.propertyListModal?.propertyPrice
            previousMarker.icon = propertyMarker.asImage()
        }
        self.propertyMapView.selectedMarker = nil

        self.propertyListContainerForMap.isHidden = false
        self.propertyViewForMapCollectionView.reloadData()
        return true
    }
    
    
    func clusterManager(_ clusterManager: GMUClusterManager, didTap clusterItem: GMUClusterItem) -> Bool {
        let newCamera = GMSCameraPosition.camera(withTarget: clusterItem.position,
                                                 zoom: propertyMapView.camera.zoom)
        let update = GMSCameraUpdate.setCamera(newCamera)
        propertyMapView.moveCamera(update)
        
        if let previousMarker = propertyMapView.selectedMarker as? CustomMarker{
            previousMarker.isSelected = !previousMarker.isSelected
            let propertyMarker = createMarkerView(isSelected: previousMarker.isSelected)
            propertyMarker.markerPriceLabel.text = previousMarker.propertyListModal?.propertyPrice
            previousMarker.icon = propertyMarker.asImage()
        }
        
        let item = clusterItem as? POIItem
        if let tappedMarker = item?.marker as? CustomMarker{
            tappedMarker.isSelected = !tappedMarker.isSelected
            let propertyMarker = createMarkerView(isSelected: tappedMarker.isSelected)
            propertyMarker.markerPriceLabel.text = tappedMarker.propertyListModal?.propertyPrice
            tappedMarker.icon = propertyMarker.asImage()
            propertyMapView.selectedMarker = tappedMarker
        }
        
        isClusterTapped = false
        self.propertyListContainerForMap.isHidden = false
        self.propertyViewForMapCollectionView.reloadData()
        return true
    }
}

我的 POItem 类

import Foundation
import GoogleMapsUtils
import GoogleMaps

class POIItem: NSObject, GMUClusterItem {
    var position: CLLocationCoordinate2D
    var name: String!
    var propertyListModal: PeropertyListModel?
    var marker: GMSMarker?
    
    init(position: CLLocationCoordinate2D, name: String, propertyListModal: PeropertyListModel) {
        self.position = position
        self.name = name
        self.propertyListModal = propertyListModal
    }
}

IconGenerator 的类

import Foundation
import GoogleMapsUtils

class ClusterIconGenerator: GMUDefaultClusterIconGenerator {
    
    override func icon(forSize size: UInt) -> UIImage {
        let image = textToImage(drawText: String(size) as NSString,
                                inImage: UIImage(named: "clusterMarker")!,
                                font: UIFont(name: "Futura-Medium", size: 12.0) ?? UIFont.systemFont(ofSize: 12))
        return image
    }
    
    private func textToImage(drawText text: NSString, inImage image: UIImage, font: UIFont) -> UIImage {
        
        UIGraphicsBeginImageContext(image.size)
        image.draw(in: CGRect(x: 0, y: 0, width: image.size.width, height: image.size.height))
        
        let textStyle = NSMutableParagraphStyle()
        textStyle.alignment = NSTextAlignment.center
        let textColor = UIColor.black
        let attributes=[
            NSAttributedString.Key.font: font,
            NSAttributedString.Key.paragraphStyle: textStyle,
            NSAttributedString.Key.foregroundColor: textColor]
        
        // vertically center (depending on font)
        let textH = font.lineHeight
        let textY = (image.size.height-textH)/2
        let textRect = CGRect(x: 0, y: textY, width: image.size.width, height: textH)
        text.draw(in: textRect.integral, withAttributes: attributes)
        let result = UIGraphicsGetImageFromCurrentImageContext()
        UIGraphicsEndImageContext()
        return result!
    }
}

以上是我到目前为止所做的代码,但不值得请你能找到我的出路如何实现在点击操作时在集群项目标记上呈现图标的解决方案。

标签: iosswiftgoogle-maps

解决方案


如果您有兴趣使用存储桶索引更改图标,我想您可以使用下面的代码来更改图像。您可以使用 buckets 索引并分别显示集群标记的图标。

let iconGenerator = GMUDefaultClusterIconGenerator(buckets: [5, 10, 15], backgroundImages: [#imageLiteral(resourceName: "clusterSelectedMarker"), #imageLiteral(resourceName: "clusterSelectedMarker"), #imageLiteral(resourceName: "clusterSelectedMarker")])
let algorithm = GMUNonHierarchicalDistanceBasedAlgorithm()
                                                clusterIconGenerator: iconGenerator)
var renderer = CustomClusterRenderer(mapView: propertyMapView, clusterIconGenerator: iconGenerator)
clusterManager = GMUClusterManager(map: propertyMapView, algorithm: algorithm,
                                           renderer: renderer)

推荐阅读