检索在sqlite数据库中插入的所有行,并在包含标签的表视图单元格中显示具有不同部分的子视图

时间:2011-12-20 10:56:27

标签: iphone uitableview sqlite

我是objective-c和sqlite的新手。我已成功使用sqlite将数据插入到表中。我的项目包含2页

添加提醒页面:用户输入数据并单击保存,这是右侧导航栏

查看提醒页面:这是用户在表格视图的单元格中查看已保存的提醒的地方,

即。第1单元的第1次提醒,第2单元的第2次提醒等等......

我使用以下代码更轻松地指定了部分数量,这意味着已保存提醒的数量等于查看提醒表中的部分数量

-(NSInteger)numberOfSectionsInTableView:(UITableView *)view
{    
    numTableRecords = -1;
    sqlite3_stmt *statement;
    if (sqlite3_open([self.databasePath UTF8String], &remindersDB) == SQLITE_OK) 
    {
        NSString *sqlStatement = [NSString stringWithFormat: @"select count(*) from reminders"];
        const char *sql = [sqlStatement cStringUsingEncoding:NSUTF8StringEncoding];
        if(sqlite3_prepare_v2(remindersDB, sql, -1, &statement, NULL) == SQLITE_OK) 
        {          
            while(sqlite3_step(statement) == SQLITE_ROW) 
            {
                numTableRecords = sqlite3_column_int(statement, 0);
            }
        }
        else 
        {
            printf("could not prepare statement: %s\n", sqlite3_errmsg(remindersDB));
        }
    }

    else 
    {
        NSLog(@"Error in Opening Database File");
    }
    sqlite3_close(remindersDB);
    NSLog(@"Number of records = %d",numTableRecords);
    return numTableRecords; 

}

根据以下链接中的建议,与我的要求非常相似:

Populate the table view sections with rows of table in sqlite database in an order

我接受了一个数组并将检索到的数据作为对象插入到数组中,如下所示:

-(void)viewWillAppear:(BOOL)animated
{

    //Retrieve the values of database
    const char *dbpath = [self.databasePath UTF8String];
    sqlite3_stmt *statement;
    if (sqlite3_open(dbpath, &remindersDB) == SQLITE_OK)
    {
        NSString *querySQL = [NSString stringWithFormat:@"SELECT * FROM reminders"];
        NSLog(@"Data = %@",querySQL);

        const char *query_stmt = [querySQL UTF8String];

        if (sqlite3_prepare_v2(self.remindersDB ,query_stmt , -1, &statement, NULL) == SQLITE_OK)
        {
            if (sqlite3_step(statement) == SQLITE_ROW)
            {
                ID = [[[NSString alloc] initWithUTF8String:(const char *) sqlite3_column_text(statement, 0)]autorelease];

                nameField = [[[NSString alloc]initWithUTF8String:(const char *) sqlite3_column_text(statement, 1)]autorelease];                    

                eventField = [[[NSString alloc]initWithUTF8String:(const char *) sqlite3_column_text(statement, 2)]autorelease];

                dateField = [[[NSString alloc]initWithUTF8String:(const char *) sqlite3_column_text(statement, 3)]autorelease];
            } 

            sqlite3_finalize(statement);
        }
        sqlite3_close(self.remindersDB);
    }
    [array addObject:ID];
    [array addObject:nameField];
    [array addObject:eventField];
    [array addObject:dateField];
    [self.theTable reloadData];
    [super viewWillAppear:YES];
}

现在在(UITableViewCell) - 索引路径的行的单元格中,我获取了4个标签,并将它们添加为子视图,并将[array objectAtIndex:indexPath.row]分配给label.text,即:

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

    //NSString *CellId = [NSString stringWithFormat:@"S%1dR%1d",indexPath.section,indexPath.row];

    UITableViewCell *cell = (UITableViewCell *)[view dequeueReusableCellWithIdentifier:nil];

    if (cell == nil)
    {
        cell = [[[UITableViewCell alloc] initWithStyle:UITableViewCellStyleValue1 reuseIdentifier:nil] autorelease];
        view.backgroundColor = [UIColor clearColor];
        cell.backgroundColor = [[UIColor alloc]initWithPatternImage:[UIImage imageNamed:@"reminderbutton.png"]];

        label1 = [[[UILabel alloc]initWithFrame:CGRectMake(26, 3, 30, 40)]autorelease];
        label1.backgroundColor = [UIColor clearColor];
        label1.textColor = [UIColor whiteColor];

        label2 = [[[UILabel alloc]initWithFrame:CGRectMake(45, 3, 100, 40)]autorelease];
        label2.backgroundColor = [UIColor clearColor];
        label2.textColor = [UIColor whiteColor];

        label3 = [[[UILabel alloc]initWithFrame:CGRectMake(119, 3, 100, 40)]autorelease];
        label3.backgroundColor = [UIColor clearColor];
        label3.textColor = [UIColor whiteColor];

        label4 = [[[UILabel alloc]initWithFrame:CGRectMake(198, 3, 120, 40)]autorelease];
        label4.backgroundColor = [UIColor clearColor];
        label4.textColor = [UIColor whiteColor];

    }
 label1.text = [array objectAtIndex:indexPath.row];
    label2.text = [array objectAtIndex:indexPath.row];
    label3.text = [array objectAtIndex:indexPath.row];
    label4.text = [array objectAtIndex:indexPath.row];

    [cell addSubview:label1];
    [cell addSubview:label2];
    [cell addSubview:label3];
    [cell addSubview:label4];

    return cell;
}

在细胞上没有显示任何内容,请帮助我解决这个问题。对此我非常不好,因为我是Objective-c的新手,如果有人提出详细的解释和解决方案,我会很高兴.Sorry如果它听起来很愚蠢的问题

提前感谢所有人:)

2 个答案:

答案 0 :(得分:7)

我明白你现在想做什么。

我建议创建一个类来保存每个提醒的数据,然后你就可以得到一个提醒数据数组。

所以我会在你的项目中添加一个新类,称之为Reminder,然后为每个字段添加一个属性。

您的提醒课程如下:

Reminder.h

#import <Foundation/Foundation.h>

@interface Reminder : NSObject

@property (nonatomic, copy) NSString *reminderId; 
@property (nonatomic, copy) NSString *name;
@property (nonatomic, copy) NSString *event;
@property (nonatomic, copy) NSString *date;

@end

并且您的reminder.m文件将是:

#import "Reminder.h"

@implementation Reminder
@synthesize reminderId, name, event, date;

-(void)dealloc
{

    self.reminderId = nil;
    self.name = nil;
    self.event = nil;
    self.date = nil;
}

@end

因此,此类只会保存每个提醒的数据。

然后在显示提醒列表的视图控制器中,您需要添加一个数组属性来保存提醒对象。

因此,请将以下属性添加到ViewController,以显示提醒列表。

@property (nonatomic, retain) NSMutableArray *reminders;

确保你也在视图控制器.m文件中合成它,为此,在视图控制器.m文件的@implementation行下添加这行代码:

@synthesize reminders;

还记得在dealloc方法中将reminders属性设置为nil以确保正确的内存清理:

self.reminders = nil;

我们将使用我们在视图控制器中创建的Reminder类,因此您还需要导入您的类,因此在视图控制器.m文件中,在文件顶部附近添加一个import语句,如下所示:

#import "Reminder.h"

现在您的视图控制器已设置好并可以使用我们创建的提醒类。

然后,您需要在视图控制器中添加一个方法,该方法将用于从数据库加载数据并填充提醒对象和提醒数组。

向视图控制器添加一个新方法,称之为loadReminders

此方法的实现将是:

-(void)loadReminders
{

    // setup the reminders array
    self.reminders = nil;
    self.reminders = [[[NSMutableArray alloc] init] autorelease];

    //Retrieve the values of database
    const char *dbpath = [self.databasePath UTF8String];
    sqlite3_stmt *statement;
    if (sqlite3_open(dbpath, &remindersDB) == SQLITE_OK)
    {
        NSString *querySQL = [NSString stringWithFormat:@"SELECT * FROM reminders"];
        NSLog(@"Data = %@",querySQL);

        const char *query_stmt = [querySQL UTF8String];

        if (sqlite3_prepare_v2(self.remindersDB ,query_stmt , -1, &statement, NULL) == SQLITE_OK)
        {
            while (sqlite3_step(statement) == SQLITE_ROW)
            {
                Reminder *loadedReminder = [[Reminder alloc] init];

                loadedReminder.reminderId = [[[NSString alloc] initWithUTF8String:(const char *) sqlite3_column_text(statement, 0)]autorelease];

                loadedReminder.name = [[[NSString alloc]initWithUTF8String:(const char *) sqlite3_column_text(statement, 1)]autorelease];                    

                loadedReminder.event = [[[NSString alloc]initWithUTF8String:(const char *) sqlite3_column_text(statement, 2)]autorelease];

                loadedReminder.date = [[[NSString alloc]initWithUTF8String:(const char *) sqlite3_column_text(statement, 3)]autorelease];

                [self.reminders addObject:loadedReminder];
                [loadedReminder release];
            } 

            sqlite3_finalize(statement);
        }
        sqlite3_close(self.remindersDB);
    }


}

所以这个方法正在做的是从数据库加载行,对于数据库中的每一行,它创建一个新的Reminder对象,使用行中的数据设置属性,然后将此提醒对象添加到提醒阵列。

注意我从原始代码中所做的更改:

while (sqlite3_step(statement) == SQLITE_ROW)

while语句将循环并从数据库中获取每一行。

每当你想要刷新数据列表时,你都会调用这个新方法(loadReminders),所以你当前正在重新加载表的代码中的任何地方,在重新加载表之前调用这个方法,这将确保你的数组提醒已准备好供重新加载时使用的表。因此,在上面的示例中,我将viewWillAppear方法替换为:

-(void)viewWillAppear:(BOOL)animated
{
    [self loadReminders];
    [self.theTable reloadData];
}

现在所有数据都已设置好并可以使用。使用提醒对象应该使tableView中的代码更容易阅读。

因此,要将数据放入表中,请使用以下内容替换当前的tableview方法。

-(NSInteger)numberOfSectionsInTableView:(UITableView *)view
{
     return [self.reminders count];
}

这将告诉表格为提醒数组中的每个提醒创建一个部分。

现在,根据您的描述(和链接),您正在做的是在每个部分创建1个单元格。因此,添加以下方法告诉tableView我们希望每个部分有1行

-(NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section
{
    return 1;
}

现在对于cellForRowAtIndexPath,请使用以下内容替换当前方法:

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

    Reminder *reminderToDisplay;
    reminderToDisplay = [self.reminders objectAtIndex:indexPath.section];

    // Now create the cell to display the reminder data:

    cell = [[[UITableViewCell alloc] initWithStyle:UITableViewCellStyleValue1 reuseIdentifier:nil] autorelease];
    view.backgroundColor = [UIColor clearColor];
    cell.backgroundColor = [[UIColor alloc]initWithPatternImage:[UIImage imageNamed:@"reminderbutton.png"]];

    label1 = [[[UILabel alloc]initWithFrame:CGRectMake(26, 3, 30, 40)]autorelease];
    label1.backgroundColor = [UIColor clearColor];
    label1.textColor = [UIColor whiteColor];

    label2 = [[[UILabel alloc]initWithFrame:CGRectMake(45, 3, 100, 40)]autorelease];
    label2.backgroundColor = [UIColor clearColor];
    label2.textColor = [UIColor whiteColor];

    label3 = [[[UILabel alloc]initWithFrame:CGRectMake(119, 3, 100, 40)]autorelease];
    label3.backgroundColor = [UIColor clearColor];
    label3.textColor = [UIColor whiteColor];

    label4 = [[[UILabel alloc]initWithFrame:CGRectMake(198, 3, 120, 40)]autorelease];
    label4.backgroundColor = [UIColor clearColor];
    label4.textColor = [UIColor whiteColor];


    // Now set the labels using our reminder object
    label1.text = reminderToDisplay.reminderId;
    label2.text = reminderToDisplay.name;
    label3.text = reminderToDisplay.event;
    label4.text = reminderToDisplay.date;


    // now add the labels to the cell
    cell.contentView = [[UIView alloc] init] autorelease]
    [cell.contentView addSubview:label1];
    [cell.contentView addSubview:label2];
    [cell.contentView addSubview:label3];
    [cell.contentView addSubview:label4];

    return cell;

}

所以在这个方法中发生了什么,首先我们得到提醒数组中当前部分的提示:

Reminder *reminderToDisplay;
reminderToDisplay = [self.reminders objectAtIndex:indexPath.section];

接下来的几行应该是熟悉的,我们正在创建单元格和标签。然后我们使用提醒对象来设置标签文本。

您现有解决方案无法正常工作的原因是因为您使用的是indexPath.Row,并且因为您为每个提醒使用了一个部分,所以indexPath.Row将始终为0,因为每个部分中只有1行。

我知道这篇文章很长,但希望能帮助你了解正在发生的事情。任何问题然后让我知道

答案 1 :(得分:1)

你应该真正创建一个自定义UITableViewCell。

但是对于快速修复答案,您可以使用以下

替换底部[cell addSubview]行
cell.contentView = [[UIView alloc] init] autorelease]
[cell.contentView addSubview:label1];
[cell.contentView addSubview:label2];
[cell.contentView addSubview:label3];
[cell.contentView addSubview:label4];