Использовать метод didSelectRowAtIndexPath или prepareForSegue для UITableView?

Я использую раскадровки, и у меня есть UITableView. У меня есть настройка segue, которая выталкивает из моей таблицы в деталь VC. Но какой метод я должен использовать для этого? Мне придется передать пару объектов в подробный вид. Но использую ли я didSelectRowAtIndex или -(void)prepareForSegue:(UIStoryboardSegue *)segue sender:(id)sender?

Ответ 1

Если вы используете prepareForSegue:sender:, у вас не будет столько изменений, если вы позже решите вызывать segue из некоторого элемента управления вне представления таблицы.

Сообщение prepareForSegue:sender: отправляется в текущий контроллер представления, поэтому я бы предложил следующее:

- (void)prepareForSegue:(UIStoryboardSegue *)segue sender:(id)sender
{
    // Assume self.view is the table view
    NSIndexPath *path = [self.tableView indexPathForSelectedRow];
    DetailObject *detail = [self detailForIndexPath:path];
    [segue.destinationViewController setDetail:detail];
}

В Swift:

override func prepareForSegue(segue: UIStoryboardSegue, sender: AnyObject?) {
    let path = self.tableView.indexPathForSelectedRow()!
    segue.destinationViewController.detail = self.detailForIndexPath(path)
}

Ответ 2

Я сделал это, и он работал

- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath{

    NSLog(@"Row Selected = %i",indexPath.row);

    [self performSegueWithIdentifier:@"testID" sender:self.view];    
}

Ответ 3

Когда отправителем является UITableViewCell, вы можете запросить UITableView запрос indexPath ячейки.

    override func prepareForSegue(segue: UIStoryboardSegue, sender: AnyObject?) {
        if let cell = sender as? UITableViewCell {
            let indexPath = self.tableView.indexPathForCell(cell)!
            assert(segue.destinationViewController.isKindOfClass(DetailViewController))
            let detailViewController = segue.destinationViewController as! DetailViewController
            detailViewController.item = self.items[indexPath.row] // like this
        }
    }

Ответ 4

если свойство tableView находится в другом классе, и вы имеете только один раздел, тогда вы можете использовать свойство tag для хранения строки ячейки, например:

- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {

     UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:@"cell"];

     cell.tag = indexPath.row;

     return cell;
}

И тогда вы можете получить к нему доступ, так как sender - это одна и та же ячейка со значением строки в теге:

- (void)prepareForSegue:(UIStoryboardSegue *)segue sender:(id)sender {

    MyDestinationViewController *destinationViewController = segue.destinationViewController;
    destinationViewController.myProperty = [tableViewElementsArray objectAtIndex:[sender tag]]; // sender will be your cell 
}

Ответ 5

self.tableView.indexPathForSelectedRow возвращает выбранную ячейку, но не передает ячейку отправителя, например, ячейка отправителя не выбрана (вспомогательное действие) или в случае множественного выбора. Лучший способ получить indexPath для segue sender:

- (void)prepareForSegue:(UIStoryboardSegue *)segue sender:(id)sender {
    __auto_type itemViewController = (id<ItemViewController>)segue.destinationViewController;
    itemViewController.senderIndexPath = [self.tableView indexPathForCell:sender];
}

В Свифте:

protocol ItemViewController {
    var senderIndexPath : IndexPath? { get set }
    var selectedIndexPaths : [IndexPath]? { get set }
}

override func prepare(for segue: UIStoryboardSegue, sender: Any?) {
    if  let cell = sender as? UITableViewCell,
        var itemViewController = segue.destination as? ItemViewController {
        itemViewController.senderIndexPath = tableView.indexPath(for: cell)
        itemViewController.selectedIndexPaths = tableView.indexPathsForSelectedRows
    }
}

Ответ 6

Если вы используете prepareForSegue:, вы можете проверить, кто является отправителем, и выполнить другой код

Например, в быстрой

override func prepareForSegue(segue: UiStoryboardSegue, sender: AnyObject?)
{
   var senderIsTableviewCell:Bool! = sender?.isKindOfClass(UITableViewCell)

   if senderIsTableviewCell
   {
       //do something
   }
}