Не могу перетащить UITableViewCell из текущей позиции при изменении порядка - PullRequest
16 голосов
/ 12 февраля 2012

Я пытаюсь сделать так, чтобы мои UITableView, поддерживаемые данными Core, имели возможность переупорядочения. После реализации всех этих делегатов и некоторой техники для упомянутых данных ядра здесь Я обнаружил странное поведение. После нажатия кнопки «Изменить» значок переупорядочения отображается очень хорошо, и я могу нажать на него, появляется тень, но когда я пытаюсь переместить ее в другой ряд, я внезапно теряю фокус, и ячейка возвращается на свое место. Кто-нибудь знает причину этой проблемы?

Вот видео, показывающее проблему

http://www.youtube.com/watch?v=ugxuLNL7BnU&feature=youtu.be

Вот мой код

- (BOOL)tableView:(UITableView *)tableView canMoveRowAtIndexPath:(NSIndexPath *)indexPath
{
    return YES;
}

in -tableView:moveRowAtIndexPath:toIndexPath Я попробовал приведенный ниже код и просто пустую реализацию, но проблема все еще существует, поэтому я не думаю, что это не проблема.

- (void)tableView:(UITableView *)tableView moveRowAtIndexPath:(NSIndexPath *)sourceIndexPath toIndexPath:(NSIndexPath *)destinationIndexPath
{
    _changeIsUserDriven = YES;

    NSUInteger fromIndex = sourceIndexPath.row;
    NSUInteger toIndex = destinationIndexPath.row;

    NSMutableArray *newsArray = [[self.fetchedResultsController fetchedObjects] mutableCopy];
    News *news = [self.fetchedResultsController.fetchedObjects objectAtIndex:fromIndex];

    [newsArray removeObject:news];
    [newsArray insertObject:news atIndex:toIndex];

    int i = 1;
    for (News *n in newsArray) {
        n.displayOrder = [NSNumber numberWithInt:i++];
    }

    [self saveContext];

    _changeIsUserDriven = NO;
}

FetchedResultController делегат

    - (void)controllerWillChangeContent:(NSFetchedResultsController *)controller
    {
        if (_changeIsUserDriven) {
            return;
        }
        [self.tableView beginUpdates];
    }




       - (void)controller:(NSFetchedResultsController *)controller didChangeSection:(id <NSFetchedResultsSectionInfo>)sectionInfo
                   atIndex:(NSUInteger)sectionIndex forChangeType:(NSFetchedResultsChangeType)type
        {
            if (_changeIsUserDriven) {
                return;
            }
            switch(type) {
                case NSFetchedResultsChangeInsert:
                    [self.tableView insertSections:[NSIndexSet indexSetWithIndex:sectionIndex] withRowAnimation:UITableViewRowAnimationFade];
                    break;

                case NSFetchedResultsChangeDelete:
                    [self.tableView deleteSections:[NSIndexSet indexSetWithIndex:sectionIndex] withRowAnimation:UITableViewRowAnimationFade];
                    break;
            }
        }


- (void)controller:(NSFetchedResultsController *)controller didChangeObject:(id)anObject
       atIndexPath:(NSIndexPath *)indexPath forChangeType:(NSFetchedResultsChangeType)type
      newIndexPath:(NSIndexPath *)newIndexPath
{
    if (_changeIsUserDriven) {
        return;
    }
    UITableView *tableView = self.tableView;

    switch(type) {
        case NSFetchedResultsChangeInsert:
            [tableView insertRowsAtIndexPaths:[NSArray arrayWithObject:newIndexPath] withRowAnimation:UITableViewRowAnimationFade];
            break;

        case NSFetchedResultsChangeDelete:
            [tableView deleteRowsAtIndexPaths:[NSArray arrayWithObject:indexPath] withRowAnimation:UITableViewRowAnimationFade];
            break;

        case NSFetchedResultsChangeUpdate:
            [self configureCell:[tableView cellForRowAtIndexPath:indexPath] atIndexPath:indexPath];
            break;

        case NSFetchedResultsChangeMove:
            [tableView deleteRowsAtIndexPaths:[NSArray arrayWithObject:indexPath] withRowAnimation:UITableViewRowAnimationFade];
            [tableView insertRowsAtIndexPaths:[NSArray arrayWithObject:newIndexPath]withRowAnimation:UITableViewRowAnimationFade];
            break;
    }
}

    - (void)controllerDidChangeContent:(NSFetchedResultsController *)controller
    {
        if (_changeIsUserDriven) {
            return;
        }
        [self.tableView endUpdates];
    }

Ответы [ 2 ]

18 голосов
/ 13 февраля 2012

Наконец-то я обнаружил, что проблема в том, что используемая мной внешняя библиотека, IIViewDeckController, после удаления устраняет проблему.

3 голосов
/ 10 апреля 2013

Это действительно связано с ViewDeckController. Я использую порт в MonoTouch и до сих пор сталкиваюсь с проблемой. Это связано (более конкретно) с UIPanGestureRecognizer. Распознаватель жестов принимает движение панорамирования и перетаскивания и полагает, что оно предназначено для себя, поэтому он отменяет касание, отправляемое в UITableView.

К сожалению, нет свойства UIPanGestureRecognizer.direction (горизонтальный / вертикальный), но здесь есть сообщение: UIPanGestureRecognizer - Только вертикальный или горизонтальный относительно того, как ограничить жест панорамирования только определенным направлением. Вам нужно будет изменить код ViewDeckController, где он добавляет UIPanGestureRecognizer, чтобы использовать версию с подклассами из вопроса выше.

Я еще не проверял это (так как сейчас 2 часа ночи, и я уже готов!), Но обновлю свой ответ, как только у меня будет время для этого.

В качестве быстрого и грязного исправления вы можете установить

panner.cancelsTouchesInView = NO;

Это позволит вам перемещать ваши элементы, однако также будет иметь ожидаемый результат - не отменять касания в представлении, поэтому, в зависимости от содержимого вашего центрального вида, это может привести к тому, что ваши пользователи нажмут кнопку, когда они просто проведите смотровую площадку в сторону.

Другой возможностью было бы установить для этого свойства значение NO, когда вы переводите таблицу в режим редактирования, и по окончании работы изменить ее на YES.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...