tags:

views:

1172

answers:

2

Hi guys,

actually I'm have a very simple problem (that's for sure) I just can't find the root cause.

The problem is, that after calling the numberOfSectionsInTableView Method to get the number of Sections (24 in my case) the method numberOfRowsInSection gets called but with the last section (23).

This causes an out of bound index when scrolling.

So, what happens? I'm receiving the data of the last section (23) and even the number of rows of the last section (23) and afterwards the sections are moving forward like 0,1,2,3,etc...

An example: I have a section with the section header "#" for special characters. In my case the last section is the "X" for all values beginning with an "X".

When I call my App, the section title has been set correctly to "#" but the data shows all the "X" values. At this point I have 31 "X" values. When I now scroll to the last of the 31 rows, the app crashes because of an out of index error.

I think everything would work correctly If the sections wouldn't start with the last one and afterwards continue with the first ones (23,0,1,2,3,4,etc). I just can't find the place where or why the section hast the value 23 on the first place.

---EDIT 2 BEGINS: initialising the array:

    alphabetArray = [[NSMutableArray alloc] init];

First I fill a dictionary like this (for all letters incl. special chars):

charDict = [[NSMutableDictionary alloc] init];

if ([charSpecialArray count] > 0)   {

    [charDict setObject:charSpecialArray forKey:@"Special"];
    [alphabetArray addObject:@"Special"];
}

if ([charAArray count] > 0) {

    [charDict setObject:charAArray forKey:@"A"];
    [alphabetArray addObject:@"A"];
}

---EDIT 2 ENDS

So here I'm just filling the first dictionary Object with all "#" values (going on with all the other values)

Then I call the numberOfSectionsInTableView method:

- (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView {
  return [charDict count];
}

here the numberOfRowsInSection goes:

// Customize the number of rows in the table view.
- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section {

NSString *key = [alphabetArray objectAtIndex:section];  
NSArray *array = [charDict objectForKey:key];
return [array count];
 }

}

and the cellForRowAtIndexPath:

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

  static NSString *CellIdentifier = @"Cell";

  UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:CellIdentifier];


if(cell == nil)
  cell = [[[UITableViewCell alloc] initWithStyle:UITableViewCellStyleSubtitle reuseIdentifier:@"MyIdentifier"] autorelease];

if ([charSpecialArray count] > 0)   {

    Char *charSpecial = (Char *)[[charSpecialArray objectAtIndex:indexPath.section] objectAtIndex:indexPath.row];

    cell.textLabel.numberOfLines = 1;
    cell.detailTextLabel.numberOfLines = 2;
    [cell.detailTextLabel setFont:[UIFont boldSystemFontOfSize: 12]];

    cell.textLabel.text = charSpecial.ta;
    cell.detailTextLabel.text = charSpecial.descriptionEN;
}

if ([charAArray count] > 0) {

    Char *charA = (Char *)[[charAArray objectAtIndex:indexPath.section] objectAtIndex:indexPath.row];

    cell.textLabel.numberOfLines = 1;           
            cell.detailTextLabel.numberOfLines = 2;
    [cell.detailTextLabel setFont:[UIFont boldSystemFontOfSize: 12]];

    cell.textLabel.text = charA.ta;
    cell.detailTextLabel.text = charA.descriptionEN;

    }

again for all characters...

Any suggestions how to fix that?

added initialization code .h:

#import <UIKit/UIKit.h>

@interface Char : NSObject {
NSString *ta;
NSString *report;
NSString *descriptionDE;
NSString *descriptionEN;
}

@property (nonatomic, retain) NSString *ta;
@property (nonatomic, retain) NSString *report;
@property (nonatomic, retain) NSString *descriptionDE;
@property (nonatomic, retain) NSString *descriptionEN;


-(id)initWithTa:(NSString *)t report:(NSString *)re descriptionDE:(NSString *)dde     descriptionEN:(NSString *)den;

@end

initialization code .m:

#import "SCode.h"

@implementation Char

@synthesize ta, report, descriptionDE, descriptionEN;

-(id)initWithTa:(NSString *)t report:(NSString *)re descriptionDE:(NSString *)dde descriptionEN:(NSString *)den {
self.ta = t;
self.report = re;
self.descriptionDE = dde;
self.descriptionEN = den;
return self;
}

- (void)dealloc {

[ta release];
[report release];
[descriptionDE release];
[descriptionEN release];
[super dealloc];
}

@end
+1  A: 

I haven't had my coffee yet but I don't see how this will even compile.

This line:

Char *charSpecial = (Char *)[charSpecial objectAtIndex:indexPath.row];

should presumably be:

Char *charSpecial = (Char *)[charsSpecial objectAtIndex:indexPath.row];

Note the "s". You have the Char type "charSpecial" when you wanted the NSArray "charsSpecial". (These aren't good variable names because they're differentiated only by a single letter buried in the middle. I suggest changing "charsSpecial" to "charSpecials" or better yet "charSpecilsArray".)

Not sure this is your error because it looks more like a posting typo. You seem to have the same problem here:

if ([charSpecial count] > 0) {

  NSDictionary *charsSpecialDict = [NSDictionary dictionaryWithObject:scodesSpecial forKey:@"Chars"];
  [listOfItems addObject:scodesSpecialDict];
}

This

When I call my App, the section title has been set correctly to "#" but the data shows all the "X" values.

Suggest that somewhere you are swapping your ordinal (zero indexed) value with your cardinal (one indexed). You're correctly querying the section names with an ordinal value but you are fetching your rows with the cardinal. Since the cardinal value is alway one more than the ordinal, you get an out of bounds error.

I think you should refractor you variable names and look at the code again.

TechZen
Hi TechZen,correct the first one was a typo error (just tried to anonymize my code a bit).could you please give me some more information about the ordinal and cardinal thing?The logic is clear, but I'm not able to assign this to my code (sorry for my bad english).Would it be possible to give me a specific code example?
Daniel
+2  A: 

You've got a big problem with your cellForRowAtIndexPath. CharsSpecial is accessed no matter what section you are in (so long as it has one item), and it may very well not have as many items as that section. This is most likely causing the crash. If your code is well written, then an out of order table shouldn't crash anything.

IndexPath has two components, section and row. In order to get the data you want out of a 2D array, you should use [[topArray objectAtIndex:indexPath.section] objectAtIndex:indexPath.row]. Likewise, numberOfRows should use [[topArray objectAtIndex:indexPath.section] count] and numberOfSections should use [topArray count]. This way, all of these methods are consistent. You will need to adapt this for dictionaries, but that should be trivial.

Once you've done that and you can scroll the table, post a new question about the order of your data, and include all the code that modifies listOfItems.

Edit:

ListOfItems seems to only hold the one dictionary, which means you don't really need it. I recommended using an array instead of a dictionary as your top level container because it will make numberOfRows simpler, but to use it, you need to replace, not encapsulate, the dictionary.

If you want to continue using the dictionary as the top level container that's OK. You should change numberOfSections to count the number of items in the dictionary rather than the listOfItems array. NumberOfRows should look something like this:

- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section {
    NSString *key = [sectionNameArray objectAtIndex:section];  
    NSArray *array = [dictionary objectForKey:key];
    return [array count];
}

Where sectionNameArray is an array of strings you use for dictionary keys (they don't have to be the actual section names).

Edit 2:

A few things:

1) alphabetArray has a memory leak, you shouldn't retain it as it's already been retained by alloc.

2) Any sections that you want to skip need to be removed from the alphabetArray. For this reason it might be better to start with an empty array and add a letter when you get to that if statement (if([charAArray count] >0) for example).

3) In cellForRow, you seem to be trying to display one item from each letter array all together in each table cell. Is this your intention? The reason you are only seeing the last value is because you keep changing the text of your label (if charAArray has any objects, it overwrites the value charSpecicalArray set). I suspect that what you really want to do is use the code I posted above to select one array, then use the indexPath row to select one item from that array, and set the labels using that one item. Is this correct?

Edit 3:

cellForRowAtIndexPath is supposed to return one cell given an indexPath, not all the cells for the table. I don't think you are modifying your letter arrays (you shouldn't be), so what you want is something like this:

NSString *key = [alphabetArray objectAtIndex:section];  
NSArray *array = [charDict objectForKey:key];
Char *myChar = (Char *)[array objectAtIndex:indexPath.row];

cell.textLabel.numberOfLines = 1;
cell.detailTextLabel.numberOfLines = 2;
[cell.detailTextLabel setFont:[UIFont boldSystemFontOfSize: 12]];

cell.textLabel.text = myChar.ta;
cell.detailTextLabel.text = myChar.descriptionEN;

This should replace all of the if([array count] >0) sections in cellForRowAtIndexPath

David Kanarek
Hi David,I changed the code like this: numberOfRowsInSection:NSDictionary *dictionary = [listOfItems objectAtIndex:section];NSArray *array = [dictionary objectForKey:@"SCodes"];return [array count];numberOfSectionsInTableView:return [listOfItems count];cellForRowAtIndexPath:SCode *searchSCodeSpecial = (SCode *)[[searchSCodesSpecial objectAtIndex:indexPath.section] objectAtIndex:indexPath.row];unfortunately I'm getting the following error:2010-02-13 17:15:59.756 SCode[425:207] *** -[NSCFDictionary objectAtIndex:]: unrecognized selector sent to instance 0x76f5bc0
Daniel
If I had to guess, I would say that charSpecial is a dictionary, and in cellForRowAtIndexPath you call objectAtIndex on it.
David Kanarek
Also, could you expand on the initialization code? It appears you add one object to a dictionary and only that dictionary to your listOfItems, so the table will only have one object. Is this correct?
David Kanarek
added the initialization code...actually what I currently try to do is:-execute a sql statement for each letter of the alphabet + special characters-each sql statement fills one array per letter-all arrays are filling then the dictionary-I'm using the dictionary to determine the number of sections and the number of rows per section.-In the cellForRowAtIndexPath method, I only want to fill for example the "a" section if there are values at all. therefore i'm checking the "a" array if it is filled at all. if its filled: fill the cells, if not, leave the section empty and don't even show it
Daniel
Ok getting closer..code got updated- I removed listOfItems- I'm still using a dictionary as toplevel container2 problems are still remaining1: in the cellForRowAtIndexPath method I would like to do the following:- If the array for all entries beginning with an "a" is filled at all, then use the initializtion Char *searchCharSpecial = (Char *)[[searchCharSpecialArray objectAtIndex:indexPath.section] objectAtIndex:indexPath.row];but Char is not capable to use objectAtIndex2: The first section with which the numberOfRowsInSection method is being entered is the last section (24 in my case)
Daniel
Hi David,first of all..omg..thank you for your support and time you've spent to this topic! I really appreciate that!I fixed now your first two points (source code is updated).To answer your third point:What I want to do is:lets say I have an array for all "a" values and an array for all "b" values. "a" values are 120, "b" values are emptySo, because the "a" array is filled, cellForRow should list all 120 values out of the "a" array under the sectionIndexTitle "a" (out of the alphabetArray).It's not that I only see the last value. It's that the last section gets filled first (24>1>2,etc)
Daniel
omg.. you're genius! it works perfectly now, even the sectionindextitles are working perfectly now...David, one more time: thank you very much for all your answers and for spending so much time helping me!
Daniel
Always happy to help and I'm glad it's working. Good luck on the project!
David Kanarek