Подтвердить что ты не робот

Переупорядочение основных данных UITableView

Я знаю, что этот вопрос задавали раньше, и я взглянул на ответ на этот вопрос. Тем не менее, я все еще запутался относительно того, как реализовать переупорядочение с помощью UITableView в проекте Core Data. Я знаю, что мне нужно иметь атрибут "displayOrder" в моем Entity для отслеживания порядка элементов, и мне нужно перечислить все объекты в полученных результатах и ​​установить их атрибут displayOrder.

В данном коде в вопросе, с которым я связан, метод делегирования представления таблицы вызывает метод, подобный этому [self FF_fetchResults];, и код для этого метода не указан, поэтому трудно сказать, что именно это.

Есть ли какой-нибудь пример кода, который демонстрирует это? Это было бы проще посмотреть, чем обмен большими кусками кода.

Спасибо

4b9b3361

Ответ 1

Я не уверен, в какой части у вас возникают проблемы (на основе комментариев)... но вот мое предложение. DisplayOrder - просто простой атрибут класса NSManagedObject. Если вы можете сохранить управляемый объект, вы сможете завершить эту функцию. Давайте сначала возьмем простой объект NSManagedObject:

@interface RowObj :  NSManagedObject  
{
}

@property (nonatomic, retain) NSString *rowDescription;
@property (nonatomic, retain) NSNumber *displayOrder;

Затем нам нужно иметь локальную копию данных, отображаемых в представлении таблицы. Я прочитал комментарии, которые вы сделали, и я не уверен, что вы используете FetchedResultsController или нет. Мое предложение состояло бы в том, чтобы начать простой и просто использовать обычный диспетчер tableview, где вы обновляете данные о строках всякий раз, когда пользователь меняет порядок отображения... а затем сохраняет заказ, когда пользователь закончил редактирование.

Интерфейс для этого tableviewcontoller будет выглядеть так:

@interface MyTableViewController : UITableViewController {
    NSMutableArray *myTableViewData;
}

@property(nonatomic,retain) NSMutableArray *myTableViewData;

@end 

Затем нам нужно загрузить данные табличного представления в метод viewWillAppear:

- (void)viewWillAppear:(BOOL)animated {
    myTableViewData = [helper getRowObjects]; // insert method call here to get the data
    self.navigationItem.leftBarButtonItem = [self editButtonItem];
}

Здесь происходит 2 вещи... (позже я объясню editButtonItem), во-первых, нам нужно получить наши данные из CoreData. Когда я должен это делать, у меня есть какой-то помощник (назовите это, как хотите), объект выполнит эту работу. Типичный метод поиска будет выглядеть следующим образом:

- (NSMutableArray*) getRowObjects{
    NSFetchRequest *request = [[NSFetchRequest alloc] init];
    NSEntityDescription *entity = [NSEntityDescription entityForName:@"RowObj" inManagedObjectContext:[self managedObjectContext]];
    [request setEntity:entity];

    NSSortDescriptor *sortDescriptor = [[NSSortDescriptor alloc] initWithKey:@"displayOrder" ascending:YES];
    NSArray *sortDescriptors = [[NSArray alloc] initWithObjects:sortDescriptor, nil];
    [request setSortDescriptors:sortDescriptors];
    [sortDescriptors release];
    [sortDescriptor release];

    NSError *error;
    NSMutableArray *mutableFetchResults = [[managedObjectContext executeFetchRequest:request error:&error] mutableCopy];
    if (mutableFetchResults == nil) {
        // Handle the error.
    }
    [request release];
    return mutableFetchResults;
}

Теперь, когда у вас есть данные, вы можете теперь ждать, пока пользователь отредактирует таблицу. Именно здесь вступает в действие [self editButtonItem]. Это встроенная функция, которая возвращает элемент кнопки панели, который переключает заголовок и связанное состояние между Edit и Done. Когда пользователь нажимает на эту кнопку, он будет вызывать метод setEditing: animated::

Чтобы обновить порядок отображения, вам необходимо переопределить метод setEditing в классе UITableViewController. Он должен выглядеть примерно так:

- (void)setEditing:(BOOL)editing animated:(BOOL)animated {
    [super setEditing:editing animated:animated];
    [myTableView setEditing:editing animated:animated];
    if(!editing) {
        int i = 0;
        for(RowObj *row in myTableViewData) {
            row.displayOrder = [NSNumber numberWithInt:i++];
        }
        [helper saveManagedObjectContext]; // basically calls [managedObjectContext save:&error];
    }
}

Нам не нужно ничего делать, когда пользователь находится в режиме редактирования... мы хотим сохранить только после нажатия кнопки "Готово". Когда пользователь перетаскивает строку в вашей таблице, вы можете обновить порядок отображения, переопределив методы canMoveRowAtIndexPath и moveRowAtIndexPath:

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

(void)tableView:(UITableView *)tableView moveRowAtIndexPath:(NSIndexPath *)sourceIndexPath toIndexPath:(NSIndexPath *)destinationIndexPath {

    RowObj *row = [myTableViewData objectAtIndex:sourceIndexPath.row];
    [myTableViewData removeObjectAtIndex:sourceIndexPath.row];
    [myTableViewData insertObject:row atIndex:destinationIndexPath.row];
}

Опять же, причина, по которой я не обновляю значение displayOrder, заключается в том, что пользователь все еще находится в режиме редактирования... мы не знаем, будет ли пользователь выполнен, и они могут даже отменить то, что они сделали не нажав кнопку "Готово".

ИЗМЕНИТЬ

Если вы хотите удалить строку, вам нужно переопределить tableView: commitEditingStyle: forRowAtIndexPath и сделать что-то вроде этого:

- (void)tableView:(UITableView *)tableView commitEditingStyle:(UITableViewCellEditingStyle)editingStyle forRowAtIndexPath:(NSIndexPath *)indexPath {
    if (editingStyle == UITableViewCellEditingStyleDelete) {
        // Delete the managed object at the given index path.
        RowObj *row = [myTableViewData objectAtIndex:indexPath.row];
        [helper deleteRow:row];

        // Update the array and table view.
        [myTableViewData removeObjectAtIndex:indexPath.row];
        [myTableView deleteRowsAtIndexPaths:[NSArray arrayWithObject:indexPath] withRowAnimation:YES];
    }   
}