【问题标题】:How to select map pin when tapping table row that associates with it?点击与之关联的表格行时如何选择地图图钉?
【发布时间】:2015-03-24 12:46:24
【问题描述】:

这里我有 2 个视图:

  • WallViewController
  • TableViewController

WallViewController 包含 MKMapView,TableViewControllerPFQueryTableViewController 的子类,显示与 WallViewControllerTableViewController 添加为 WallViewController 中的子视图,因此这两个视图一起显示。 现在当我在地图上点击注释引脚时, 中的 tableview TableViewController 将滚动到与成功选择的注释引脚关联的行。 但我想做的是也有一个相反的动作,所以当我点击一行时,与该行关联的注释图钉将被选中并显示其标注。

下面我提供了相关代码:

WallViewController.h

@interface WallViewController : UIViewController <MKMapViewDelegate, CLLocationManagerDelegate>

@property (nonatomic, strong) IBOutlet MKMapView *mapView;
@end

@protocol WallViewControllerHighlight <NSObject>
- (void)highlightCellForPost:(PAWPost *)post;
- (void)unhighlightCellForPost:(PAWPost *)post;
@end

WallViewController.m

- (void)viewDidLoad {
    [super viewDidLoad];
    self.tableViewController = [[TableViewController alloc] initWithStyle:UITableViewStylePlain];
    [self addChildViewController:self.tableViewController];

    self.tableViewController.view.frame = CGRectMake(6.0f, 215.0f, 308.0f, self.view.bounds.size.height - 215.0f);
    [self.view addSubview:self.tableViewController.view];
    self.wallPostsTableViewController.wallViewController = self;
    ....
    }

- (void)mapView:(MKMapView *)mapView didSelectAnnotationView:(MKAnnotationView *)view {
    id<MKAnnotation> annotation = [view annotation];
    if ([annotation isKindOfClass:[PAWPost class]]) {
        PAWPost *post = [view annotation];                        
        [PostsTableViewController highlightCellForPost:post];
    } else if ([annotation isKindOfClass:[MKUserLocation class]]) {
        // Center the map on the user's current location:
        AppDelegate *appDelegate = [[UIApplication sharedApplication] delegate];
        MKCoordinateRegion newRegion = MKCoordinateRegionMakeWithDistance(appDelegate.currentLocation.coordinate, appDelegate.filterDistance * 2, appDelegate.filterDistance * 2);

        [self.mapView setRegion:newRegion animated:YES];
        self.mapPannedSinceLocationUpdate = NO;
    }
}

TableViewController.h

@interface TableViewController : PFQueryTableViewController <WallViewControllerHighlight>

@property (nonatomic, strong)  PAWWallViewController *wallViewController;
- (void)highlightCellForPost:(PAWPost *)post;
- (void)unhighlightCellForPost:(PAWPost *)post;
@end

TableViewController.m

....
@synthesize wallViewController;
....

- (void)highlightCellForPost:(PAWPost *)post {
    // Find the cell matching this object.
    for (PFObject *object in [self objects]) {
        PAWPost *postFromObject = [[PAWPost alloc] initWithPFObject:object];
        if ([post equalToPost:postFromObject]) {
            // We found the object, scroll to the cell position where this object is.
            NSUInteger index = [[self objects] indexOfObject:object];

            NSIndexPath *indexPath = [NSIndexPath indexPathForRow:index inSection:kTableViewMainSection];
            [self.tableView scrollToRowAtIndexPath:indexPath atScrollPosition:UITableViewScrollPositionMiddle animated:YES];
            [self.tableView selectRowAtIndexPath:indexPath animated:NO scrollPosition:UITableViewScrollPositionNone];

            return;
        }
    }

    // Don't scroll for posts outside the search radius.
    if ([post.title compare:kWallCantViewPost] != NSOrderedSame) {
        // We couldn't find the post, so scroll down to the load more cell.
        NSUInteger rows = [self.tableView numberOfRowsInSection:kTableViewMainSection];
        NSIndexPath *indexPath = [NSIndexPath indexPathForRow:(rows - 1) inSection:kTableViewMainSection];
        [self.tableView scrollToRowAtIndexPath:indexPath atScrollPosition:UITableViewScrollPositionBottom animated:YES];
        [self.tableView selectRowAtIndexPath:indexPath animated:NO scrollPosition:UITableViewScrollPositionNone];
        [self.tableView deselectRowAtIndexPath:indexPath animated:YES];
    }
}

- (void)unhighlightCellForPost:(PAWPost *)post {
    // Deselect the post's row.
    for (PFObject *object in [self objects]) {
        PAWPost *postFromObject = [[PAWPost alloc] initWithPFObject:object];
        if ([post equalToPost:postFromObject]) {
            NSUInteger index = [[self objects] indexOfObject:object];
            NSIndexPath *indexPath = [NSIndexPath indexPathForRow:index inSection:0];
            [self.tableView deselectRowAtIndexPath:indexPath animated:YES];

            return;
        }
    }
}

- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath {
    // call super because we're a custom subclass.
    [super tableView:tableView didSelectRowAtIndexPath:indexPath];

    [tableView deselectRowAtIndexPath:indexPath animated:YES];

    // highlight annotation pin on map - my problem is here ????

    NSLog(@"tap row --> %i", indexPath.row); //******1*****
    NSLog(@"annotation count --> %i", [self.objects count]); //*****2*****
    UITableViewCell *selectedCell = [tableView cellForRowAtIndexPath:indexPath];
    UILabel *textLabel = (UILabel*) [selectedCell.contentView viewWithTag:kPAWCellTextLabelTag];
    NSLog(@"textlabel at the table row tapped --> %@", textLabel.text); //*****3*****


    for (NSInteger i = 0; i < [self.objects count]; i++) {

        PFObject *object = [self.objects objectAtIndex:i];
        PAWPost *postFromObject = [[PAWPost alloc] initWithPFObject:object];
        NSString *text = postFromObject.title;
        NSLog(@"text retrieved from PAWPost --> %@", text); //*****4*****

        if ([text isEqualToString:textLabel.text]) {
            NSLog(@"found matching at %i", i); //*****5*****
            [wallViewController.mapView selectAnnotation:[[wallViewController.mapView annotations] objectAtIndex:i] animated:YES];
            NSLog(@"wallViewController --> %@", wallViewController); //*****6*****
            NSLog(@"wallViewController mapView --> %@", wallViewController.mapView); //*****7*****
        }

    }
}

我的问题在于 TableViewController.m 中的最后一个方法 (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath。现在我有你看到的代码,使用:

[wallViewController.mapView selectAnnotation:[[wallViewController.mapView annotations] objectAtIndex:i] animated:YES];

但它似乎不起作用。这里一定有问题,但我不知道是什么。

点击表格的第一行后,我在这里提供了我的日志输出:

2015-01-28 11:24:31.886 Test[8097:60b] tap row --> 0
2015-01-28 11:24:31.888 Test[8097:60b] annotation count --> 40
2015-01-28 11:24:31.890 Test[8097:60b] textlabel at the table row tapped --> Asddsdtest
2015-01-28 11:24:31.890 Test[8097:60b] text retrieved from PAWPost --> ...(40 texts here)
2015-01-28 11:24:31.897 Test[8097:60b] found matching at 0
2015-01-28 11:24:31.899 Test[8097:60b] wallViewController --> (null)
2015-01-28 11:24:31.900 Test[8097:60b] wallViewController mapView --> (null)

附:我找到了this question,但答案和 cmets 似乎没有为像我这样的初学者提供足够的细节。

添加了其他信息

PAWPost.h

@interface PAWPost : NSObject <MKAnnotation>

//@protocol MKAnnotation <NSObject>

// Center latitude and longitude of the annotion view.
// The implementation of this property must be KVO compliant.
@property (nonatomic, readonly) CLLocationCoordinate2D coordinate;

// @optional
// Title and subtitle for use by selection UI.
@property (nonatomic, readonly, copy) NSString *title;
@property (nonatomic, readonly, copy) NSString *subtitle;
// @end

// Other properties:
@property (nonatomic, readonly, strong) PFObject *object;
@property (nonatomic, readonly, strong) PFGeoPoint *geopoint;
@property (nonatomic, readonly, strong) PFUser *user;
@property (nonatomic, assign) BOOL animatesDrop;
@property (nonatomic, readonly) MKPinAnnotationColor pinColor;

// Designated initializer.
- (id)initWithCoordinate:(CLLocationCoordinate2D)coordinate andTitle:(NSString   *)title andSubtitle:(NSString *)subtitle;
- (id)initWithPFObject:(PFObject *)object;
- (BOOL)equalToPost:(PAWPost *)aPost;

- (void)setTitleAndSubtitleOutsideDistance:(BOOL)outside;

@end

PAWPost.m

@interface PAWPost ()

// Redefine these properties to make them read/write for internal class accesses and mutations.
@property (nonatomic, assign) CLLocationCoordinate2D coordinate;

@property (nonatomic, copy) NSString *title;
@property (nonatomic, copy) NSString *subtitle;

@property (nonatomic, strong) PFObject *object;
@property (nonatomic, strong) PFGeoPoint *geopoint;
@property (nonatomic, strong) PFUser *user;
@property (nonatomic, assign) MKPinAnnotationColor pinColor;

@end

@implementation PAWPost

- (id)initWithCoordinate:(CLLocationCoordinate2D)aCoordinate andTitle:(NSString *)aTitle andSubtitle:(NSString *)aSubtitle {
    self = [super init];
    if (self) {
        self.coordinate = aCoordinate;
        self.title = aTitle;
        self.subtitle = aSubtitle;
        self.animatesDrop = NO;
    }
    return self;
}

- (id)initWithPFObject:(PFObject *)anObject {
    self.object = anObject;
    self.geopoint = [anObject objectForKey:kPAWParseLocationKey];
    self.user = [anObject objectForKey:kPAWParseUserKey];

    [anObject fetchIfNeeded]; 
    CLLocationCoordinate2D aCoordinate = CLLocationCoordinate2DMake(self.geopoint.latitude, self.geopoint.longitude);
    NSString *aTitle = [anObject objectForKey:kPAWParseTextKey];
    NSString *aSubtitle = [[anObject objectForKey:kPAWParseUserKey] objectForKey:kPAWParseUsernameKey];

    return [self initWithCoordinate:aCoordinate andTitle:aTitle andSubtitle:aSubtitle];
}

- (BOOL)equalToPost:(PAWPost *)aPost {
    if (aPost == nil) {
        return NO;
   }

    if (aPost.object && self.object) {
        // We have a PFObject inside the PAWPost, use that instead.
        if ([aPost.object.objectId compare:self.object.objectId] != NSOrderedSame) {
            return NO;
        }
        return YES;
    } else {
        // Fallback code:

        if ([aPost.title compare:self.title] != NSOrderedSame ||
        [aPost.subtitle compare:self.subtitle] != NSOrderedSame ||
        aPost.coordinate.latitude != self.coordinate.latitude ||
        aPost.coordinate.longitude != self.coordinate.longitude ) {
            return NO;
        }

        return YES;
    }
}

- (void)setTitleAndSubtitleOutsideDistance:(BOOL)outside {
    if (outside) {
        self.subtitle = nil;
        self.title = kPAWWallCantViewPost;
        self.pinColor = MKPinAnnotationColorRed;
    } else {
        self.title = [self.object objectForKey:kPAWParseTextKey];
        //self.subtitle = [[self.object objectForKey:kPAWParseUserKey] objectForKey:kPAWParseUsernameKey];
        self.subtitle = [NSString stringWithFormat:@"@%@",[[self.object objectForKey:kPAWParseUserKey] objectForKey:kPAWParseUsernameKey]];
        self.pinColor = MKPinAnnotationColorGreen;
    }
}

@end

【问题讨论】:

  • WallVC 中的注解类型为PAWPost。 TableVC 中的行类型为PFObject。在didSelectRowAtIndexPath 中,您需要遍历地图视图的annotations 数组并找到哪个(PAWPost)与选定的PFObject(即[self.objects objectAtIndex:indexPath.row])匹配?不确定您的objects 数组与表视图部分/行)。找到匹配的注解后,您可以使用它调用 selectAnnotation(您可能需要先将地图居中放置以确保它可见,否则 selectAnnotation 将不执行任何操作)。
  • @Anna 非常感谢您的参与。我正在尝试按照您的建议进行操作,并且很难将我的编码逻辑正确地做到这一点,但没有成功。请在 (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath 处查看我更新的 TableViewController.m
  • 把那个长的、嵌套的语句分成多个,这样更容易调试。检查每一步的变量值(使用 NSLogs)。 wallViewController 是零吗? wallViewController.mapView 为零吗?等等。单步调试器中的代码。 selectAnnotation 行真的被执行了吗?确保代码在做你认为它在做的事情。
  • 为什么要创建一个新的 WallVC 实例?您的问题是“TableVC 被添加为 WallVC 中的子视图”。因此,WallVC 实例已经存在,只需将 TableVC 中的 wallViewController 属性设置为指向该现有实例即可。当 TableVC 创建时(由 WallVC?),设置它的 wallViewController 属性。
  • 我认为您不清楚一些基础知识,例如类与实例以及变量声明与设置变量值之间的区别。 WallVC 似乎是父对象,因此 TableVC 不应创建 WallVC 的新实例。相反,在 WallVC viewDidLoad 中,在创建 TableVC 之后,执行self.tableViewController.wallViewController = self;。在 TableVC 中,只需进行属性声明——不要创建新的 WallVC。

标签: ios objective-c parse-platform mkmapview tableview


【解决方案1】:

你可以使用 PFObject.objectId 来找到你想要的注解。

PFObject *selectedObject = [self.objects objectAtIndex:i];

for (id annotation in wallViewController.mapView.annotations)
{
    if ([annotation isKindOfClass:[PAWPost class]])
    {
        PAWPost *post = (PAWPost*)annotation;
        if ([post.object.objectId isEqualToString:selectedObject.objectId])
        {
            [wallViewController.mapView selectAnnotation:annotation  animated:YES];
            break;
        }
    }
}

【讨论】:

  • 非常感谢,但它不起作用。我发现 For 循环 永远不会运行。我真的怀疑我的 wallViewController.map 设置不正确,但我不知道出了什么问题。我已经在 TableViewController.m 中设置了 wallViewController 的属性,并对其进行了合成。我还缺少什么?
  • 原来是一些基本的东西乱七八糟导致这不起作用。它已整理好,您的代码运行良好。使用 objectId 查找注释对我来说是完美的。非常感谢。
【解决方案2】:

您可以将注释作为支持 MSAnnotation 协议的对象。 因此,当您选择您的单元格时 - 您将确切知道此操作背后的对象是什么。 在这种情况下,您的注释将完全是您通过点击 UITableViewCell 选择的对象。

所以 - 你的对象 PAWPost 应该支持 MKAnnotation 协议:

@interface PAWPost : NSObject <MKAnnotation> 
// Interface here
@end

@implementation PAWPost
// ...
- (CLLocationCoordinate2D)coordinate {
    return <#your coordinate here#>;
}
// ...
@end

附: 您不需要在 didSelectRowAtIndexPath: 方法中创建 WallViewController 类的新本地实例。 因为那时你的 WallViewController 不会有 MapView。 取而代之的是 - 在之前创建 WallViewController 的实例并使用指向它的指针。

【讨论】:

  • 非常感谢@WinDMak3r。如果您不介意,您能否将您的答案也写在代码中,以便我更好地理解?抱歉请求,但我是初学者,最好尝试从示例中学习?请多多包涵。
  • 我还是不明白。我应该如何处理你的代码?
  • 将它与您的 PAWPost 类一起使用以支持注释。当您在 tableView 中选择单元格时,您将使用 indexPath.row 从数组中获取此对象,然后在 selectAnnotation: 方法中使用它。
  • 不好意思说我还是不明白,但我永远不会放弃。当然,我必须让自己对协议的东西有足够的了解。顺便说一句,我添加了我现在得到的 PAWPost 类的详细信息。最初我做了这样的事情 [wallViewController.mapView selectAnnotation:[[wallViewController.mapView annotations] objectAtIndex:indexPath.row] animated:YES];。我知道这行不通。我只是不知道如何将注释与 indexPath.row 链接。如果您编辑我的代码来实现这一点,那将是非常有帮助的。
  • 在您的 tableViewController 中,您需要添加另一个属性并将其与现有的 MapViewController 连接。之后你可以轻松地做这样的事情:[wallViewController.mapView selectAnnotation:self.objects[indexPath.row] animated:YES]
猜你喜欢
  • 1970-01-01
  • 2014-10-17
  • 2015-09-05
  • 2013-03-23
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多