在 iOS 中创建可编辑的表格视图?

Posted

技术标签:

【中文标题】在 iOS 中创建可编辑的表格视图?【英文标题】:Creating an editable Table View in iOS? 【发布时间】:2013-05-13 18:14:56 【问题描述】:

我正在尝试使用 .xib 文件在 ios 中创建一个可编辑的表格视图

我的代码如下所示:

viewController.h

#import <UIKit/UIKit.h>

@interface ViewController : UITableViewController

@property (nonatomic, strong) NSMutableArray *notes;

@end

viewController.m

#import "ViewController.h"
#import "Note.h"

@interface ViewController ()

@end

@implementation ViewController

- (id)initWithStyle:(UITableViewStyle)style

    self = [super initWithStyle:style];
    if (self) 
        // Custom initialization
    
    return self;


- (void)viewDidLoad

    [super viewDidLoad];

    [self setEditing:NO animated:NO];


- (void)viewWillAppear:(BOOL)animated

    self.notes = [[Note savedNotes]mutableCopy];
    [self.tableView reloadData];


- (void)didReceiveMemoryWarning

    [super didReceiveMemoryWarning];
    // Dispose of any resources that can be recreated.


#pragma mark - Button Events

- (void)setEditing:(BOOL)editing animated:(BOOL)animated

    [super setEditing:editing animated:animated];

    if (editing) 
        UIBarButtonItem *cancelButton = [[UIBarButtonItem alloc]initWithBarButtonSystemItem:UIBarButtonSystemItemCancel target:self action:@selector(cancelButtonPressed:)];
        self.navigationItem.leftBarButtonItem = cancelButton;

        UIBarButtonItem *doneButton = [[UIBarButtonItem alloc]initWithBarButtonSystemItem:UIBarButtonSystemItemDone target:self action:@selector(rightButtonPressed:)];

        self.navigationItem.rightBarButtonItem = doneButton;

    else
        self.navigationItem.leftBarButtonItem = [[UIBarButtonItem alloc]initWithBarButtonSystemItem:UIBarButtonSystemItemAdd
                                                                                             target:self
                                                                                             action:@selector(addNoteButtonPressed:)];

        UIBarButtonItem *editButton = [[UIBarButtonItem alloc]initWithBarButtonSystemItem:UIBarButtonSystemItemEdit
                                                                                   target:self
                                                                                   action:@selector(rightButtonPressed:)];

        self.navigationItem.rightBarButtonItem = editButton;
    



- (void)rightButtonPressed:(id)sender

    [self setEditing:!self.isEditing animated:YES];


- (void)cancelButtonPressed:(id)sender
    [self setEditing:!self.isEditing animated:YES];


- (void)addNoteButtonPressed:(id)sender
    //ATTViewController *viewController = [self.storyboard instantiateViewControllerWithIdentifier:@"ATTViewController"];
    //[self.navigationController pushViewController:viewController animated:YES];


#pragma mark - Table view data source

- (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView

    // Return the number of sections.
    return 1;


- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section

    // Return the number of rows in the section.
    return [self.notes count];


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

    static NSString *CellIdentifier = @"Cell";
    UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:CellIdentifier forIndexPath:indexPath];

    // Configure the cell...

    Note *note = self.notes[indexPath.row];

    cell.textLabel.text = note.name;
    cell.detailTextLabel.text = note.event;

    return cell;



// Override to support conditional editing of the table view.
- (BOOL)tableView:(UITableView *)tableView canEditRowAtIndexPath:(NSIndexPath *)indexPath

    // Return NO if you do not want the specified item to be editable.
    return YES;


- (UITableViewCellEditingStyle)tableView:(UITableView *)tableView editingStyleForRowAtIndexPath:(NSIndexPath *)indexPath

    return UITableViewCellEditingStyleDelete;



// Override to support editing the table view.
- (void)tableView:(UITableView *)tableView commitEditingStyle:(UITableViewCellEditingStyle)editingStyle forRowAtIndexPath:(NSIndexPath *)indexPath

    if (editingStyle == UITableViewCellEditingStyleDelete) 
        // Delete the row from the data source
        Note *note = self.notes[indexPath.row];
        [note remove];
        [self.notes removeObjectAtIndex:indexPath.row];
        [tableView deleteRowsAtIndexPaths:@[indexPath]
                         withRowAnimation:UITableViewRowAnimationFade];
    
    else if (editingStyle == UITableViewCellEditingStyleInsert) 
        // Create a new instance of the appropriate class, insert it into the array, and add a new row to the table view
    



/*
 // Override to support rearranging the table view.
 - (void)tableView:(UITableView *)tableView moveRowAtIndexPath:(NSIndexPath *)fromIndexPath toIndexPath:(NSIndexPath *)toIndexPath
 
 
 */

/*
 // Override to support conditional rearranging of the table view.
 - (BOOL)tableView:(UITableView *)tableView canMoveRowAtIndexPath:(NSIndexPath *)indexPath
 
 // Return NO if you do not want the item to be re-orderable.
 return YES;
 
 */

#pragma mark - Table view delegate

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

    // Navigation logic may go here. Create and push another view controller.
    /*
     <#DetailViewController#> *detailViewController = [[<#DetailViewController#> alloc] initWithNibName:@"<#Nib name#>" bundle:nil];
     // ...
     // Pass the selected object to the new view controller.
     [self.navigationController pushViewController:detailViewController animated:YES];
     */
    [tableView deselectRowAtIndexPath:indexPath animated:YES];


@end

在 .xib 的身份检查器中,我将类从 UIView 更改为 UITableView

我将视图与数据源和/或委托连接(我尝试了所有组合)。

但是,我在电话上总是一无所获。我没有得到表格视图,只是一个空的灰色屏幕。

任何想法我做错了什么?上面的代码在一个使用故事板的项目中工作,现在我正试图让它与 .xib 文件一起工作。

截图:

【问题讨论】:

notes 数组是否包含任何数据? tableView 至少需要返回一行。向数组中添加一个对象并进行测试。 在应用程序的第一次启动中,表格视图是空的。这并不意味着认为它不应该被加载。如您所见,应该加载一个可编辑的表格视图、一个栏、一个编辑按钮、一个添加按钮等。我不明白您的第二点,这似乎更重要。你能解释一下你的意思吗? 它可能已加载,但没有返回单元格 忽略第二部分,因为您使用的是 tableViewController。对于那个很抱歉。 tableView 是否分配给 xib 中的任何内容?委托需要指向您的班级。 我将 xib 分配给委托人 【参考方案1】:

我建议查看免费的 Sensible TableView 框架。将自动处理显示您的数组并代表您处理所有插入/删除。为我节省了大量时间。

【讨论】:

【参考方案2】:

经过一些工作,我能够做到并决定将其公开:

Code on GitHub

我正在学习如何做到这一点,但我想支持移动、删除和添加。

一些示例仅显示最后一行的添加按钮,而其他示例仅显示移动/删除。

对我来说棘手的一点是同时支持 3 个。

希望对你有帮助。

代码如下:

EditTVC.h

#import <UIKit/UIKit.h>

@interface EditTVC : UITableViewController

@end

EditTVC.m

//
//  EditTVC.m
//  editTableViewTest
//

#import "EditTVC.h"

@interface EditTVC ()

@property (nonatomic) NSMutableArray *myItems;

@end

@implementation EditTVC

@synthesize myItems;

- (void)viewDidLoad

    [super viewDidLoad];

    myItems = [[NSMutableArray alloc] init];

    for (int i=0; i<5; i++)
    
        [myItems addObject:[NSString stringWithFormat:@"Hey %d",i]];
    

    NSLog(@"%@",myItems);


    // Display an Edit button in the navigation bar for this view controller.
    self.navigationItem.rightBarButtonItem = self.editButtonItem;


#pragma mark - Table view data source


- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section

    // Return the number of rows in the section.

    if (self.editing) 
        return myItems.count +1;
    

    return myItems.count;


-(UITableViewCellEditingStyle)tableView:(UITableView *)tableView editingStyleForRowAtIndexPath:(NSIndexPath *)indexPath 
    if (indexPath.row == myItems.count)
        return UITableViewCellEditingStyleInsert;
    else
        return UITableViewCellEditingStyleDelete;


- (BOOL)tableView:(UITableView *)tableView canMoveRowAtIndexPath:(NSIndexPath *)indexPath

    // Return NO if you do not want the item to be re-orderable.
    if (indexPath.row == myItems.count)
        return NO;
    else
        return YES;


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

    static NSString *CellIdentifier = @"MyCell";
    UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:CellIdentifier forIndexPath:indexPath];

    // Configure the cell...

    if (self.editing && indexPath.row == myItems.count)
    
        cell.textLabel.text = @"Add ...";
    
    else
    
        cell.textLabel.text = myItems[indexPath.row];
    

    return cell;


-(void)setEditing:(BOOL)editing animated:(BOOL)animated 

    [super setEditing:editing animated:animated];

    if(editing)
    
        //edit mode
    

    else
    
        //non-edit mode
    

    [self.tableView reloadData];


// Override to support editing the table view.
- (void)tableView:(UITableView *)tableView commitEditingStyle:(UITableViewCellEditingStyle)editingStyle forRowAtIndexPath:(NSIndexPath *)indexPath

    if (editingStyle == UITableViewCellEditingStyleDelete)
    
        // Delete the row from the data source
        [myItems removeObjectAtIndex:indexPath.row];

        [tableView deleteRowsAtIndexPaths:@[indexPath] withRowAnimation:UITableViewRowAnimationFade];
       
    else if (editingStyle == UITableViewCellEditingStyleInsert)
    
        // Create a new instance of the appropriate class, insert it into the array, and add a new row to the table view
        [myItems insertObject:[NSString stringWithFormat:@"Added %ld",(long)indexPath.row] atIndex:indexPath.row];
        [tableView insertRowsAtIndexPaths:@[indexPath] withRowAnimation:UITableViewRowAnimationFade];
    
    NSLog(@"%@",myItems);


// Override to support rearranging the table view.
- (void)tableView:(UITableView *)tableView moveRowAtIndexPath:(NSIndexPath *)fromIndexPath toIndexPath:(NSIndexPath *)toIndexPath

    if (toIndexPath.row == myItems.count) //we only check the toIndexPath because we made the AddCell not to respond to move events
    
        id tmp = [myItems objectAtIndex:fromIndexPath.row];
        [myItems removeObjectAtIndex:fromIndexPath.row];
        [myItems insertObject:tmp atIndex:toIndexPath.row-1]; //to keep it in valid range for the NSMutableArray

        [self.tableView reloadData];
    

    else
    
        id tmp = [myItems objectAtIndex:fromIndexPath.row];
        [myItems removeObjectAtIndex:fromIndexPath.row];
        [myItems insertObject:tmp atIndex:toIndexPath.row];
    

    NSLog(@"%@",myItems);




@end

【讨论】:

请在此处发布代码,而不是提供代码链接。谢谢 :) 我在答案中添加了代码(除了 GitHub 存储库)@SaghirA.Khatri【参考方案3】:

将至少一项添加到您的数组中,否则将不显示任何内容(当前您的 cellForRowAtIndexPath 正在返回 0)。另外,请确保 tableView 的委托已分配给您在 IB 中的类。

【讨论】:

你错了。使用故事板,我可以看到一个空的表格视图。现在我什么也没看到。就像没有加载一样。我无法与应用程序交互。应该有一个添加按钮,用户可以在其中添加 smth 到表视图等,但现在什么都没有加载。一开始表格视图应该是空的 添加了一个截图,这样你就可以看到连接 在告诉帮助你的人“你错了”之前,请先听听我们在说什么。首先,在您的屏幕截图中,您没有将数据源和委托设置为文件所有者。先做这个。其次,添加一个视图然后将tableView添加到视图中......

以上是关于在 iOS 中创建可编辑的表格视图?的主要内容,如果未能解决你的问题,请参考以下文章

如何在 ios 中创建可扩展的表格视图?

Kendo UI - 如何在编辑时使特定字段只读,同时在剑道网格中创建可编辑?

如何在 HTML/Javascript 中创建可编辑的组合框?

如何在HTML / Javascript中创建可编辑的组合框?

在 Google AdWords 中创建可重用函数

在 Swift 中创建可重用视图并添加完成处理程序