Ответ 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];
}
}