Skip to content

Commit dd422cb

Browse files
committed
Refactored how sorting of collections works
1 parent 7a6215c commit dd422cb

13 files changed

+725
-39
lines changed

.gitignore

+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
/.DS_Store

MAPIClient.h

+1-5
Original file line numberDiff line numberDiff line change
@@ -10,11 +10,9 @@
1010

1111
#define NOTIF_USER_CHANGED @"m_user_changed"
1212
#define NOTIF_COLLECTION_CHANGED @"m_collection_changed"
13+
#define NOTIF_MODEL_CHANGED @"m_model_changed"
1314
#define NOTIF_API_QUEUE_CHANGED @"m_api_queue_changed"
1415

15-
#define API_TIMESTAMP_FORMAT @"yyyy-MM-dd'T'HH:mm:ss.SSS'Z'"
16-
17-
1816
@interface MAPIClient : AFHTTPClient
1917
{
2018
NSMutableArray * _transactionsQueue;
@@ -37,8 +35,6 @@
3735
- (void)getCollectionAtPath:(NSString*)path userTriggered:(BOOL)triggered success:(void (^)(id responseObject))successCallback failure:(void (^)(NSError *err))failureCallback;
3836
- (void)requestPath:(NSString*)path withMethod:(NSString*)method withParameters: params userTriggered:(BOOL)triggered expectedClass:(Class)expectation success:(void (^)(id responseObject))successCallback failure:(void (^)(NSError *err))failureCallback;
3937

40-
- (void)authenticateWithToken:(NSString*)accessToken;
41-
4238

4339
#pragma mark Tracking API Access and Recovering from Offline State
4440

MAPIClient.m

+4-1
Original file line numberDiff line numberDiff line change
@@ -19,7 +19,7 @@ + (MAPIClient *)shared
1919
static MAPIClient * sharedClient = nil;
2020
static dispatch_once_t onceToken;
2121
dispatch_once(&onceToken, ^{
22-
sharedClient = [[MAPIClient alloc] initWithBaseURL:[NSURL URLWithString: BASE_URL]];
22+
sharedClient = [[MAPIClient alloc] initWithBaseURL:[NSURL URLWithString: API_ROOT]];
2323
});
2424
return sharedClient;
2525
}
@@ -236,6 +236,9 @@ - (void)criticalRequestFailed:(NSError*)err
236236
NSDictionary * json = [NSJSONSerialization JSONObjectWithData:jsonData options:NSJSONReadingAllowFragments error:NULL];
237237
if (json) message = [json objectForKey: @"error"];
238238
}
239+
240+
if ([err code] == 401)
241+
message = @"Please check your email address and password.";
239242

240243
if (message)
241244
[[[UIAlertView alloc] initWithTitle:@"Error" message:message delegate:nil cancelButtonTitle:@"OK" otherButtonTitles:nil] show];

MModel.h

+2
Original file line numberDiff line numberDiff line change
@@ -37,6 +37,8 @@ typedef enum SaveState {
3737

3838
- (void)setup;
3939

40+
- (NSComparisonResult)sort:(MModel*)other;
41+
4042
- (NSString*)ID;
4143
- (BOOL)isSaved;
4244
- (BOOL)isUnsaved;

MModel.m

+10-1
Original file line numberDiff line numberDiff line change
@@ -60,6 +60,11 @@ - (void)setup
6060

6161
}
6262

63+
- (NSComparisonResult)sort:(MModel*)other
64+
{
65+
return [[self createdAt] compare: [other createdAt]];
66+
}
67+
6368
- (BOOL)isSaved
6469
{
6570
return [self ID] != nil;
@@ -134,14 +139,16 @@ - (void)updateWithResourceJSON:(NSDictionary*)json
134139

135140
} else if ([type isEqualToString: @"T@\"MModelCollection\""]) {
136141
MModelCollection * collection = (MModelCollection *)*value;
137-
[collection updateWithResourceJSON: [json objectForKey: jsonKey]];
142+
[collection updateWithResourceJSON: [json objectForKey: jsonKey] discardMissingModels: YES];
138143
[collection setRefreshDate: [NSDate date]];
139144

140145
} else {
141146
*value = [json objectForKey: jsonKey];
142147
}
143148
return YES;
144149
}];
150+
151+
[[NSNotificationCenter defaultCenter] postNotificationName:NOTIF_MODEL_CHANGED object:self];
145152
}
146153

147154
- (void)save:(MAPITransactionCallback)callback
@@ -204,6 +211,8 @@ - (void)setEachPropertyInSet:(NSArray*)properties withValueProvider:(BOOL (^)(id
204211
} else {
205212
if ([type isEqualToString: @"T@\"NSString\""] && [value isKindOfClass: [NSNumber class]])
206213
value = [(NSNumber*)value stringValue];
214+
if ([type isEqualToString: @"Tc"] && [value isKindOfClass: [NSString class]])
215+
value = [NSNumber numberWithChar: [(NSString*)value characterAtIndex: 0]];
207216
[self setValue:value forKey:key];
208217
}
209218
}

MModelCollection.h

+4-3
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,7 @@
1515
@interface MModelCollection : NSObject <NSCoding, MRestfulObject>
1616
{
1717
NSMutableArray * _cache;
18+
BOOL _loadReturnedZero;
1819
}
1920

2021
@property (nonatomic, strong) NSString * collectionName;
@@ -42,15 +43,15 @@
4243
- (void)removeItemAtIndex:(NSUInteger)i;
4344
- (void)removeItemWithID:(NSString*)ID;
4445

45-
- (void)updateWithResourceJSON:(NSArray*)jsons;
46+
- (void)updateWithResourceJSON:(NSArray*)jsons discardMissingModels:(BOOL)discardMissing;
47+
- (void)updateFromPath:(NSString*)path replaceExistingContents:(BOOL)replace;
4648

4749
- (NSArray*)all;
4850
- (int)count;
4951

5052
- (void)refresh;
5153
- (void)refreshIfOld;
5254

53-
54-
55+
- (void)loadMore;
5556

5657
@end

MModelCollection.m

+43-29
Original file line numberDiff line numberDiff line change
@@ -119,12 +119,15 @@ - (void)removeItemWithID:(NSString*)ID
119119
[self removeItemAtIndex: [[self all] indexOfObject: [self objectWithID: ID]]];
120120
}
121121

122-
- (void)updateWithResourceJSON:(NSArray*)jsons
122+
- (void)updateWithResourceJSON:(NSArray*)jsons discardMissingModels:(BOOL)discardMissing
123123
{
124124
NSMutableArray * unused = [NSMutableArray arrayWithArray: _cache];
125125

126126
for (NSDictionary * json in jsons) {
127-
NSString * ID = [[json objectForKey: @"id"] stringValue];
127+
NSString * ID = [json objectForKey: @"id"];
128+
if ([ID isKindOfClass: [NSString class]] == NO)
129+
ID = [(NSNumber*)ID stringValue];
130+
128131
MModel * existing = nil;
129132
for (MModel * obj in unused) {
130133
if ([[obj ID] isEqualToString: ID]) {
@@ -143,59 +146,70 @@ - (void)updateWithResourceJSON:(NSArray*)jsons
143146
}
144147
}
145148

146-
[unused makeObjectsPerformSelector:@selector(setParent:) withObject: nil];
147-
[_cache removeObjectsInArray: unused];
148-
149+
if (discardMissing) {
150+
[unused makeObjectsPerformSelector:@selector(setParent:) withObject: nil];
151+
[_cache removeObjectsInArray: unused];
152+
}
149153

150-
// sort the items using their createdAt date
151-
// TODO REVISIT AND MAKE A MODEL METHOD
152-
// [__cache sortUsingComparator:^NSComparisonResult(id obj1, id obj2) {
153-
// if ([obj1 displayOrder] < [obj2 displayOrder])
154-
// return NSOrderedAscending;
155-
// else if ([obj1 displayOrder] > [obj2 displayOrder])
156-
// return NSOrderedDescending;
157-
// return NSOrderedSame;
158-
// }];
159-
}
160-
161-
- (NSArray*)all
162-
{
163-
[self refreshIfOld];
164-
return _cache;
154+
[_cache sortUsingSelector: @selector(sort:)];
165155
}
166156

167-
- (int)count
168-
{
169-
return [[self all] count];
170-
}
171-
172-
- (void)refresh
157+
- (void)updateFromPath:(NSString*)path replaceExistingContents:(BOOL)replace
173158
{
174159
if (_refreshInProgress)
175160
return;
176161

177162
_refreshInProgress = YES;
178163

179-
[[MAPIClient shared] getCollectionAtPath:[self resourcePath] userTriggered:NO success:^(id responseObject) {
180-
[self updateWithResourceJSON: responseObject];
164+
[[MAPIClient shared] getCollectionAtPath:path userTriggered:NO success:^(id responseObject) {
165+
_loadReturnedZero = ([responseObject count] == 0);
166+
[self updateWithResourceJSON: responseObject discardMissingModels: replace];
181167
[self setRefreshDate: [NSDate date]];
182168
[[MAPIClient shared] updateDiskCache: NO];
183169
_refreshInProgress = NO;
184170
[[NSNotificationCenter defaultCenter] postNotificationName:NOTIF_COLLECTION_CHANGED object:self];
185-
171+
186172
} failure:^(NSError *err) {
187173
[self setRefreshDate: [NSDate date]];
188174
_refreshInProgress = NO;
175+
_loadReturnedZero = NO;
189176
[[NSNotificationCenter defaultCenter] postNotificationName:NOTIF_COLLECTION_CHANGED object:self];
190177
}];
191178
}
192179

180+
181+
- (NSArray*)all
182+
{
183+
[self refreshIfOld];
184+
return _cache;
185+
}
186+
187+
- (int)count
188+
{
189+
return [[self all] count];
190+
}
191+
192+
- (void)refresh
193+
{
194+
[self updateFromPath:[self resourcePath] replaceExistingContents:YES];
195+
}
196+
193197
- (void)refreshIfOld
194198
{
195199
BOOL expired = (!_refreshDate || ([_refreshDate timeIntervalSinceNow] > 5000));
196200
if (expired)
197201
[self refresh];
198202
}
199203

204+
- (void)loadMore
205+
{
206+
if (_loadReturnedZero)
207+
return;
208+
209+
int page = floorf([[self all] count] / 10.0) + 1;
210+
NSString * path = [[self resourcePath] stringByAppendingFormat:@"?page=%d", page];
211+
[self updateFromPath: path replaceExistingContents:NO];
212+
}
213+
200214

201215
@end

NSContainers+NullHandlers.h

+32
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,32 @@
1+
//
2+
// NSDictionary+NullHandlers.h
3+
// Wannado
4+
//
5+
// Copyright 2013 PAR Works, Inc.
6+
//
7+
// Licensed under the Apache License, Version 2.0 (the "License");
8+
// you may not use this file except in compliance with the License.
9+
// You may obtain a copy of the License at
10+
//
11+
// http://www.apache.org/licenses/LICENSE-2.0
12+
//
13+
// Unless required by applicable law or agreed to in writing, software
14+
// distributed under the License is distributed on an "AS IS" BASIS,
15+
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
16+
// See the License for the specific language governing permissions and
17+
// limitations under the License.
18+
//
19+
20+
#import <Foundation/Foundation.h>
21+
22+
@interface NSDictionary (NullHandlers)
23+
24+
- (id)objectForKey:(id)aKey or:(id)ifNull;
25+
26+
@end
27+
28+
@interface NSArray (NullHandlers)
29+
30+
- (id)objectAtIndex:(NSUInteger)index or:(id)ifNullOrOutOfBounds;
31+
32+
@end

NSContainers+NullHandlers.m

+48
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,48 @@
1+
//
2+
// NSDictionary+NullHandlers.m
3+
// Wannado
4+
//
5+
// Copyright 2013 PAR Works, Inc.
6+
//
7+
// Licensed under the Apache License, Version 2.0 (the "License");
8+
// you may not use this file except in compliance with the License.
9+
// You may obtain a copy of the License at
10+
//
11+
// http://www.apache.org/licenses/LICENSE-2.0
12+
//
13+
// Unless required by applicable law or agreed to in writing, software
14+
// distributed under the License is distributed on an "AS IS" BASIS,
15+
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
16+
// See the License for the specific language governing permissions and
17+
// limitations under the License.
18+
//
19+
20+
#import "NSContainers+NullHandlers.h"
21+
22+
@implementation NSDictionary (NullHandlers)
23+
24+
- (id)objectForKey:(id)aKey or:(id)ifNull
25+
{
26+
id o = [self objectForKey: aKey];
27+
if ((o == nil) || ([o isKindOfClass: [NSNull class]]))
28+
return ifNull;
29+
return o;
30+
}
31+
32+
@end
33+
34+
@implementation NSArray (NullHandlers)
35+
36+
- (id)objectAtIndex:(NSUInteger)index or:(id)ifNullOrOutOfBounds
37+
{
38+
if (index >= [self count])
39+
return ifNullOrOutOfBounds;
40+
41+
id o = [self objectAtIndex: index];
42+
if ([o isKindOfClass: [NSNull class]])
43+
return ifNullOrOutOfBounds;
44+
45+
return o;
46+
}
47+
48+
@end

NSObject+AssociatedObjects.h

+3
Original file line numberDiff line numberDiff line change
@@ -11,4 +11,7 @@
1111
- (void)associateValue:(id)value withKey:(void *)key; // Strong reference
1212
- (void)weaklyAssociateValue:(id)value withKey:(void *)key;
1313
- (id)associatedValueForKey:(void *)key;
14+
15+
- (void)performSelectorOnMainThreadOnce:(SEL)selector;
16+
1417
@end

NSObject+AssociatedObjects.m

+12
Original file line numberDiff line numberDiff line change
@@ -25,4 +25,16 @@ - (id)associatedValueForKey:(void *)key
2525
return objc_getAssociatedObject(self, key);
2626
}
2727

28+
- (void)performSelectorOnMainThreadOnce:(SEL)selector
29+
{
30+
[self associateValue:[NSNumber numberWithBool: YES] withKey: (void*)selector];
31+
32+
dispatch_async(dispatch_get_main_queue(), ^{
33+
if ([self associatedValueForKey: (void*)selector]) {
34+
[self performSelector: selector];
35+
[self associateValue:nil withKey:(void*)selector];
36+
}
37+
});
38+
}
39+
2840
@end

0 commit comments

Comments
 (0)