Merge pull request #5 from mapswithme/bugs

Bugs
This commit is contained in:
deathbaba 2014-01-23 02:34:22 -08:00
commit aea874ef5e
11 changed files with 3655 additions and 594 deletions

3
.gitignore vendored Normal file
View file

@ -0,0 +1,3 @@
capitals-example/Capitals.xcodeproj/project.xcworkspace/xcshareddata/
capitals-example/Capitals.xcodeproj/project.xcworkspace/xcuserdata/
capitals-example/Capitals.xcodeproj/xcuserdata/

View file

@ -28,45 +28,47 @@
#import <Foundation/Foundation.h>
#if __IPHONE_OS_VERSION_MIN_REQUIRED < __IPHONE_4_3
#error "MapsWithMe supports iOS >= 4.3 only"
#if __IPHONE_OS_VERSION_MIN_REQUIRED < __IPHONE_5_0
#error "MapsWithMe supports iOS >= 5.0 only"
#endif
// Wrapper for a pin on a map
@interface MWMPin : NSObject
// [required] pin latitude and longitude
/// [required] pin latitude
@property (nonatomic, assign) double lat;
/// [required] pin longitude
@property (nonatomic, assign) double lon;
// [optional] pin title
/// [optional] pin title
@property (nonatomic, retain) NSString * title;
// [optional] passed back to the app when pin is clicked, OR, if it's a valid url,
// it will be opened from MapsWithMe after selecting "More Details..." for the pin
/// [optional] passed back to the app when pin is clicked, OR, if it's a valid url,
/// it will be opened from MapsWithMe after selecting "More Details..." for the pin
@property (nonatomic, retain) NSString * idOrUrl;
- (id) initWithLat:(double)lat lon:(double)lon title:(NSString *)title andId:(NSString *)idOrUrl;
- (id)initWithLat:(double)lat lon:(double)lon title:(NSString *)title andId:(NSString *)idOrUrl;
@end
// MapsWithMe API interface
@interface MWMApi : NSObject
// returns YES if url is received from MapsWithMe and can be parsed
+ (BOOL) isMapsWithMeUrl:(NSURL *)url;
// returns nil if user didn't select any pin and simply pressed "Back" button
+ (MWMPin *) pinFromUrl:(NSURL *)url;
// returns NO if MapsWithMe is not installed or outdated version doesn't support API calls
+ (BOOL) isApiSupported;
// Simply opens MapsWithMe app
+ (BOOL) showMap;
// Displays given point on a map, title and id are optional
// If id contains valid url, it will be opened from MapsWithMe after selecting "More Details..." for the pin
+ (BOOL) showLat:(double)lat lon:(double)lon title:(NSString *)title andId:(NSString *)idOrUrl;
// The same as above but using pin wrapper
+ (BOOL) showPin:(MWMPin *)pin;
// Displays any number of pins
+ (BOOL) showPins:(NSArray *)pins;
/// returns YES if url is received from MapsWithMe and can be parsed
+ (BOOL)isMapsWithMeUrl:(NSURL *)url;
/// returns nil if user didn't select any pin and simply pressed "Back" button
+ (MWMPin *)pinFromUrl:(NSURL *)url;
/// returns NO if MapsWithMe is not installed or outdated version doesn't support API calls
+ (BOOL)isApiSupported;
/// Simply opens MapsWithMe app
+ (BOOL)showMap;
/// Displays given point on a map, title and id are optional.
/// If id contains valid url, it will be opened from MapsWithMe after selecting "More Details..." for the pin
+ (BOOL)showLat:(double)lat lon:(double)lon title:(NSString *)title andId:(NSString *)idOrUrl;
/// The same as above but using pin wrapper
+ (BOOL)showPin:(MWMPin *)pin;
/// Displays any number of pins
+ (BOOL)showPins:(NSArray *)pins;
//
+ (void) showMapsWithMeIsNotInstalledDialog;
// Set value = YES if you want to open pin URL on balloon click, default value is NO
+(void) setOpenUrlOnBalloonClick:(BOOL)value;
+ (void)showMapsWithMeIsNotInstalledDialog;
/// Set value = YES if you want to open pin URL on balloon click, default value is NO
+ (void)setOpenUrlOnBalloonClick:(BOOL)value;
@end

View file

@ -28,14 +28,14 @@
#import "MapsWithMeAPI.h"
#define MAPSWITHME_API_VERSION 1
#define MAPSWITHME_API_VERSION 1.1
static NSString * MWMUrlScheme = @"mapswithme://";
static BOOL openUrlOnBalloonClick = NO;
@implementation MWMPin
- (id) init
- (id)init
{
if ((self = [super init]))
{
@ -45,7 +45,7 @@ static BOOL openUrlOnBalloonClick = NO;
return self;
}
- (id) initWithLat:(double)lat lon:(double)lon title:(NSString *)title andId:(NSString *)idOrUrl
- (id)initWithLat:(double)lat lon:(double)lon title:(NSString *)title andId:(NSString *)idOrUrl
{
if ((self = [super init]))
{
@ -57,159 +57,14 @@ static BOOL openUrlOnBalloonClick = NO;
return self;
}
- (void)dealloc
{
self.title = nil;
self.idOrUrl = nil;
[super dealloc];
}
@end
// Utility class to automatically handle "MapsWithMe is not installed" situations
@interface MWMNavigationController : UINavigationController
@end
@implementation MWMNavigationController
- (void)onCloseButtonClicked:(id)sender
{
[self dismissModalViewControllerAnimated:YES];
}
@interface MWMNViewController : UIViewController <UIWebViewDelegate>
@end
@implementation MWMApi
// Escape special chars with percent encoding
+ (NSString *) percentEncode:(NSString *)str
{
CFStringRef cfStr = (CFStringRef)str;
CFStringRef cfEncodedStr = CFURLCreateStringByAddingPercentEscapes(kCFAllocatorDefault, cfStr, NULL, CFSTR("&?/:="), kCFStringEncodingUTF8);
NSString * encodedStr = [[(NSString *)cfEncodedStr retain] autorelease];
CFRelease(cfEncodedStr);
return encodedStr;
}
+ (BOOL) isMapsWithMeUrl:(NSURL *)url
{
NSString * appScheme = [MWMApi detectBackUrlScheme];
return appScheme && [url.scheme isEqualToString:appScheme];
}
+ (MWMPin *) pinFromUrl:(NSURL *)url
{
if (![MWMApi isMapsWithMeUrl:url])
return nil;
MWMPin * pin = nil;
if ([url.host isEqualToString:@"pin"])
{
pin = [[[MWMPin alloc] init] autorelease];
for (NSString * param in [url.query componentsSeparatedByString:@"&"])
{
NSArray * values = [param componentsSeparatedByString:@"="];
if (values.count == 2)
{
NSString * key = [values objectAtIndex:0];
if ([key isEqualToString:@"ll"])
{
NSArray * coords = [[values objectAtIndex:1] componentsSeparatedByString:@","];
if (coords.count == 2)
{
pin.lat = [[NSDecimalNumber decimalNumberWithString:[coords objectAtIndex:0]] doubleValue];
pin.lon = [[NSDecimalNumber decimalNumberWithString:[coords objectAtIndex:1]] doubleValue];
}
}
else if ([key isEqualToString:@"n"])
pin.title = [[values objectAtIndex:1] stringByReplacingPercentEscapesUsingEncoding:NSUTF8StringEncoding];
else if ([key isEqualToString:@"id"])
pin.idOrUrl = [[values objectAtIndex:1] stringByReplacingPercentEscapesUsingEncoding:NSUTF8StringEncoding];
else
NSLog(@"Unsupported url parameters: %@", values);
}
}
// do not accept invalid coordinates
if (pin.lat > 90. || pin.lat < -90. || pin.lon > 180. || pin.lon < -180.)
pin = nil;
}
return pin;
}
+ (BOOL) isApiSupported
{
return [[UIApplication sharedApplication] canOpenURL:[NSURL URLWithString:MWMUrlScheme]];
}
+ (BOOL) showMap
{
return [[UIApplication sharedApplication] openURL:[NSURL URLWithString:[MWMUrlScheme stringByAppendingFormat:@"map?v=%d", MAPSWITHME_API_VERSION]]];
}
+ (BOOL) showLat:(double)lat lon:(double)lon title:(NSString *)title andId:(NSString *)idOrUrl
{
MWMPin * pin = [[[MWMPin alloc] initWithLat:lat lon:lon title:title andId:idOrUrl] autorelease];
return [MWMApi showPin:pin];
}
+ (BOOL) showPin:(MWMPin *)pin
{
return [MWMApi showPins:[NSArray arrayWithObject:pin]];
}
+ (BOOL) showPins:(NSArray *)pins
{
// Automatic check that MapsWithMe is installed
if (![MWMApi isApiSupported])
{
// Display dialog with link to the app
[MWMApi showMapsWithMeIsNotInstalledDialog];
return NO;
}
NSString * appName = [[NSBundle mainBundle] objectForInfoDictionaryKey:@"CFBundleDisplayName"];
NSMutableString * str = [[NSMutableString alloc] initWithFormat:@"%@map?v=%d&appname=%@&", MWMUrlScheme, MAPSWITHME_API_VERSION,
[MWMApi percentEncode:appName]];
NSString * backUrlScheme = [MWMApi detectBackUrlScheme];
if (backUrlScheme)
[str appendFormat:@"backurl=%@&", [MWMApi percentEncode:backUrlScheme]];
for (MWMPin * point in pins)
{
[str appendFormat:@"ll=%f,%f&", point.lat, point.lon];
@autoreleasepool
{
if (point.title)
[str appendFormat:@"n=%@&", [MWMApi percentEncode:point.title]];
if (point.idOrUrl)
[str appendFormat:@"id=%@&", [MWMApi percentEncode:point.idOrUrl]];
}
}
if (openUrlOnBalloonClick)
[str appendString:@"&balloonAction=openUrlOnBalloonClick"];
NSURL * url = [[NSURL alloc] initWithString:str];
[str release];
BOOL const result = [[UIApplication sharedApplication] openURL:url];
[url release];
return result;
}
+ (NSString *) detectBackUrlScheme
{
for (NSDictionary * dict in [[NSBundle mainBundle] objectForInfoDictionaryKey:@"CFBundleURLTypes"])
{
if ([[dict objectForKey:@"CFBundleURLName"] rangeOfString:@"mapswithme" options:NSCaseInsensitiveSearch].location != NSNotFound)
{
for (NSString * scheme in [dict objectForKey:@"CFBundleURLSchemes"])
{
// We use the first scheme in this list, you can change this behavior if needed
return scheme;
}
}
}
NSLog(@"WARNING: No com.mapswithme.maps url schemes are added in the Info.plist file. Please add them if you want API users to come back to your app.");
return nil;
}
@implementation MWMNViewController
// HTML page for users who didn't install MapsWithMe
static NSString * mapsWithMeIsNotInstalledPage =
@ -236,28 +91,164 @@ static NSString * mapsWithMeIsNotInstalledPage =
"</body>" \
"</html>";
// For gethostbyname below
#include <netdb.h>
+ (void) showMapsWithMeIsNotInstalledDialog
- (void)webView:(UIWebView *)webView didFailLoadWithError:(NSError *)error
{
UIWebView * webView = [[[UIWebView alloc] initWithFrame:[[UIScreen mainScreen] applicationFrame]] autorelease];
// check that we have Internet connection and display fresh online page if possible
if (gethostbyname("mapswith.me"))
[webView loadRequest:[NSURLRequest requestWithURL:[NSURL URLWithString:@"http://mapswith.me/api_mwm_not_installed"]]];
else
[webView loadHTMLString:mapsWithMeIsNotInstalledPage baseURL:[NSURL URLWithString:@"http://mapswith.me/"]];
UIViewController * webController = [[[UIViewController alloc] init] autorelease];
webController.view = webView;
webController.title = @"Install MapsWithMe";
MWMNavigationController * navController = [[[MWMNavigationController alloc] initWithRootViewController:webController] autorelease];
navController.navigationBar.topItem.rightBarButtonItem = [[[UIBarButtonItem alloc] initWithTitle:@"Close" style:UIBarButtonItemStyleDone target:navController action:@selector(onCloseButtonClicked:)] autorelease];
[[[UIApplication sharedApplication] delegate].window.rootViewController presentModalViewController:navController animated:YES];
[(UIWebView *)self.view loadHTMLString:mapsWithMeIsNotInstalledPage baseURL:[NSURL URLWithString:@"http://mapswith.me/"]];
}
+(void) setOpenUrlOnBalloonClick:(BOOL)value
- (void)onCloseButtonClicked:(id)sender
{
[self dismissViewControllerAnimated:YES completion:nil];
}
@end
@implementation MWMApi
+ (NSString *)urlEncode:(NSString *)str
{
return (__bridge_transfer NSString *)CFURLCreateStringByAddingPercentEscapes(kCFAllocatorDefault, (CFStringRef)str, NULL, CFSTR("!$&'()*+,-./:;=?@_~"), kCFStringEncodingUTF8);
}
+ (BOOL)isMapsWithMeUrl:(NSURL *)url
{
NSString * appScheme = [MWMApi detectBackUrlScheme];
return appScheme && [url.scheme isEqualToString:appScheme];
}
+ (MWMPin *)pinFromUrl:(NSURL *)url
{
if (![MWMApi isMapsWithMeUrl:url])
return nil;
MWMPin * pin = nil;
if ([url.host isEqualToString:@"pin"])
{
pin = [[MWMPin alloc] init];
for (NSString * param in [url.query componentsSeparatedByString:@"&"])
{
NSArray * values = [param componentsSeparatedByString:@"="];
if ([values count] == 2)
{
NSString * key = values[0];
if ([key isEqualToString:@"ll"])
{
NSArray * coords = [values[1] componentsSeparatedByString:@","];
if ([coords count] == 2)
{
pin.lat = [[NSDecimalNumber decimalNumberWithString:coords[0]] doubleValue];
pin.lon = [[NSDecimalNumber decimalNumberWithString:coords[1]] doubleValue];
}
}
else if ([key isEqualToString:@"n"])
pin.title = [values[1] stringByReplacingPercentEscapesUsingEncoding:NSUTF8StringEncoding];
else if ([key isEqualToString:@"id"])
pin.idOrUrl = [values[1] stringByReplacingPercentEscapesUsingEncoding:NSUTF8StringEncoding];
else
NSLog(@"Unsupported url parameters: %@", values);
}
}
// do not accept invalid coordinates
if (pin.lat > 90. || pin.lat < -90. || pin.lon > 180. || pin.lon < -180.)
pin = nil;
}
return pin;
}
+ (BOOL)isApiSupported
{
return [[UIApplication sharedApplication] canOpenURL:[NSURL URLWithString:MWMUrlScheme]];
}
+ (BOOL)showMap
{
return [[UIApplication sharedApplication] openURL:[NSURL URLWithString:[MWMUrlScheme stringByAppendingFormat:@"map?v=%f", MAPSWITHME_API_VERSION]]];
}
+ (BOOL)showLat:(double)lat lon:(double)lon title:(NSString *)title andId:(NSString *)idOrUrl
{
MWMPin * pin = [[MWMPin alloc] initWithLat:lat lon:lon title:title andId:idOrUrl];
return [MWMApi showPin:pin];
}
+ (BOOL)showPin:(MWMPin *)pin
{
return [MWMApi showPins:@[pin]];
}
+ (BOOL)showPins:(NSArray *)pins
{
// Automatic check that MapsWithMe is installed
if (![MWMApi isApiSupported])
{
// Display dialog with link to the app
[MWMApi showMapsWithMeIsNotInstalledDialog];
return NO;
}
NSString * appName = [[NSBundle mainBundle] objectForInfoDictionaryKey:@"CFBundleDisplayName"];
NSMutableString * str = [[NSMutableString alloc] initWithFormat:@"%@map?v=%f&appname=%@&", MWMUrlScheme, MAPSWITHME_API_VERSION,
[self urlEncode:appName]];
NSString * backUrlScheme = [MWMApi detectBackUrlScheme];
if (backUrlScheme)
[str appendFormat:@"backurl=%@&", [self urlEncode:backUrlScheme]];
for (MWMPin * point in pins)
{
[str appendFormat:@"ll=%f,%f&", point.lat, point.lon];
@autoreleasepool
{
if (point.title)
[str appendFormat:@"n=%@&", [self urlEncode:point.title]];
if (point.idOrUrl)
[str appendFormat:@"id=%@&", [self urlEncode:point.idOrUrl]];
}
}
if (openUrlOnBalloonClick)
[str appendString:@"&balloonAction=openUrlOnBalloonClick"];
NSURL * url = [NSURL URLWithString:str];
BOOL const result = [[UIApplication sharedApplication] openURL:url];
return result;
}
+ (NSString *)detectBackUrlScheme
{
for (NSDictionary * dict in [[NSBundle mainBundle] objectForInfoDictionaryKey:@"CFBundleURLTypes"])
{
if ([dict[@"CFBundleURLName"] rangeOfString:@"mapswithme" options:NSCaseInsensitiveSearch].location != NSNotFound)
{
for (NSString * scheme in dict[@"CFBundleURLSchemes"])
{
// We use the first scheme in this list, you can change this behavior if needed
return scheme;
}
}
}
NSLog(@"WARNING: No com.mapswithme.maps url schemes are added in the Info.plist file. Please add them if you want API users to come back to your app.");
return nil;
}
+ (void)showMapsWithMeIsNotInstalledDialog
{
UIWebView * webView = [[UIWebView alloc] initWithFrame:[UIScreen mainScreen].applicationFrame];
// check that we have Internet connection and display fresh online page if possible
[webView loadRequest:[NSURLRequest requestWithURL:[NSURL URLWithString:@"http://mapswith.me/api_mwm_not_installed"]]];
MWMNViewController * webController = [[MWMNViewController alloc] init];
webView.delegate = webController;
webController.view = webView;
webController.title = @"Install MapsWithMe";
UINavigationController * navController = [[UINavigationController alloc] initWithRootViewController:webController];
navController.navigationBar.topItem.rightBarButtonItem = [[UIBarButtonItem alloc] initWithTitle:@"Close" style:UIBarButtonItemStyleDone target:webController action:@selector(onCloseButtonClicked:)];
UIWindow * window = [[UIApplication sharedApplication].windows firstObject];
[window.rootViewController presentModalViewController:navController animated:YES];
}
+ (void)setOpenUrlOnBalloonClick:(BOOL)value
{
openUrlOnBalloonClick = value;
}

View file

@ -7,6 +7,7 @@
objects = {
/* Begin PBXBuildFile section */
97F3580C185F5BB200DDF84D /* capitals.plist in Resources */ = {isa = PBXBuildFile; fileRef = 97F3580B185F5BB200DDF84D /* capitals.plist */; };
FA1792CE17784F000092B567 /* MapsWithMeAPI.m in Sources */ = {isa = PBXBuildFile; fileRef = FA1792CC17784F000092B567 /* MapsWithMeAPI.m */; };
FA776B4F17848A370023F7A0 /* MasterViewController.xib in Resources */ = {isa = PBXBuildFile; fileRef = FA776B4D17848A370023F7A0 /* MasterViewController.xib */; };
FAA484EA178108970027B232 /* 114x114.png in Resources */ = {isa = PBXBuildFile; fileRef = FAA484E2178108970027B232 /* 114x114.png */; };
@ -26,12 +27,12 @@
FAD3DD63177221B500B0735B /* Default@2x.png in Resources */ = {isa = PBXBuildFile; fileRef = FAD3DD62177221B500B0735B /* Default@2x.png */; };
FAD3DD65177221B500B0735B /* Default-568h@2x.png in Resources */ = {isa = PBXBuildFile; fileRef = FAD3DD64177221B500B0735B /* Default-568h@2x.png */; };
FAD3DD68177221B500B0735B /* MasterViewController.m in Sources */ = {isa = PBXBuildFile; fileRef = FAD3DD67177221B500B0735B /* MasterViewController.m */; };
FAD3DD7F1772246600B0735B /* City.m in Sources */ = {isa = PBXBuildFile; fileRef = FAD3DD7E1772246600B0735B /* City.m */; };
FAD3DD8317724D4A00B0735B /* CityDetailViewController.m in Sources */ = {isa = PBXBuildFile; fileRef = FAD3DD8117724D4A00B0735B /* CityDetailViewController.m */; };
FAD3DD8417724D4A00B0735B /* CityDetailViewController.xib in Resources */ = {isa = PBXBuildFile; fileRef = FAD3DD8217724D4A00B0735B /* CityDetailViewController.xib */; };
/* End PBXBuildFile section */
/* Begin PBXFileReference section */
97F3580B185F5BB200DDF84D /* capitals.plist */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.plist.xml; path = capitals.plist; sourceTree = "<group>"; };
FA1792CC17784F000092B567 /* MapsWithMeAPI.m */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.objc; name = MapsWithMeAPI.m; path = ../api/MapsWithMeAPI.m; sourceTree = "<group>"; };
FA1792CD17784F000092B567 /* MapsWithMeAPI.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = MapsWithMeAPI.h; path = ../api/MapsWithMeAPI.h; sourceTree = "<group>"; };
FA776B4D17848A370023F7A0 /* MasterViewController.xib */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = file.xib; path = MasterViewController.xib; sourceTree = "<group>"; };
@ -57,8 +58,6 @@
FAD3DD64177221B500B0735B /* Default-568h@2x.png */ = {isa = PBXFileReference; lastKnownFileType = image.png; path = "Default-568h@2x.png"; sourceTree = "<group>"; };
FAD3DD66177221B500B0735B /* MasterViewController.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = MasterViewController.h; sourceTree = "<group>"; };
FAD3DD67177221B500B0735B /* MasterViewController.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = MasterViewController.m; sourceTree = "<group>"; };
FAD3DD7D177223BB00B0735B /* City.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; path = City.h; sourceTree = "<group>"; };
FAD3DD7E1772246600B0735B /* City.m */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.objc; path = City.m; sourceTree = "<group>"; };
FAD3DD8017724D4A00B0735B /* CityDetailViewController.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; path = CityDetailViewController.h; sourceTree = "<group>"; };
FAD3DD8117724D4A00B0735B /* CityDetailViewController.m */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.objc; path = CityDetailViewController.m; sourceTree = "<group>"; };
FAD3DD8217724D4A00B0735B /* CityDetailViewController.xib */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = file.xib; path = CityDetailViewController.xib; sourceTree = "<group>"; };
@ -120,8 +119,6 @@
children = (
FAD3DD5D177221B500B0735B /* AppDelegate.h */,
FAD3DD5E177221B500B0735B /* AppDelegate.m */,
FAD3DD7D177223BB00B0735B /* City.h */,
FAD3DD7E1772246600B0735B /* City.m */,
FAD3DD8017724D4A00B0735B /* CityDetailViewController.h */,
FAD3DD8117724D4A00B0735B /* CityDetailViewController.m */,
FAD3DD8217724D4A00B0735B /* CityDetailViewController.xib */,
@ -136,6 +133,7 @@
FAD3DD55177221B500B0735B /* Supporting Files */ = {
isa = PBXGroup;
children = (
97F3580B185F5BB200DDF84D /* capitals.plist */,
FAA484E2178108970027B232 /* 114x114.png */,
FAA484E3178108970027B232 /* 144x144.png */,
FAA484E4178108970027B232 /* 100x100.png */,
@ -206,6 +204,7 @@
buildActionMask = 2147483647;
files = (
FAD3DD61177221B500B0735B /* Default.png in Resources */,
97F3580C185F5BB200DDF84D /* capitals.plist in Resources */,
FAD3DD63177221B500B0735B /* Default@2x.png in Resources */,
FAD3DD65177221B500B0735B /* Default-568h@2x.png in Resources */,
FAD3DD8417724D4A00B0735B /* CityDetailViewController.xib in Resources */,
@ -231,7 +230,6 @@
FAD3DD5B177221B500B0735B /* main.m in Sources */,
FAD3DD5F177221B500B0735B /* AppDelegate.m in Sources */,
FAD3DD68177221B500B0735B /* MasterViewController.m in Sources */,
FAD3DD7F1772246600B0735B /* City.m in Sources */,
FAD3DD8317724D4A00B0735B /* CityDetailViewController.m in Sources */,
FA1792CE17784F000092B567 /* MapsWithMeAPI.m in Sources */,
);
@ -268,10 +266,11 @@
FA776B5217848EC50023F7A0 /* Production */ = {
isa = XCBuildConfiguration;
buildSettings = {
CLANG_ENABLE_OBJC_ARC = YES;
GCC_PRECOMPILE_PREFIX_HEADER = YES;
GCC_PREFIX_HEADER = "Capitals/Capitals-Prefix.pch";
INFOPLIST_FILE = "Capitals/Capitals-Info.plist";
IPHONEOS_DEPLOYMENT_TARGET = 4.3;
IPHONEOS_DEPLOYMENT_TARGET = 5.0;
PRODUCT_NAME = "World Capitals";
WRAPPER_EXTENSION = app;
};
@ -334,10 +333,11 @@
FAD3DD7B177221B500B0735B /* Debug */ = {
isa = XCBuildConfiguration;
buildSettings = {
CLANG_ENABLE_OBJC_ARC = YES;
GCC_PRECOMPILE_PREFIX_HEADER = YES;
GCC_PREFIX_HEADER = "Capitals/Capitals-Prefix.pch";
INFOPLIST_FILE = "Capitals/Capitals-Info.plist";
IPHONEOS_DEPLOYMENT_TARGET = 4.3;
IPHONEOS_DEPLOYMENT_TARGET = 5.0;
PRODUCT_NAME = "World Capitals";
WRAPPER_EXTENSION = app;
};
@ -346,10 +346,11 @@
FAD3DD7C177221B500B0735B /* Release */ = {
isa = XCBuildConfiguration;
buildSettings = {
CLANG_ENABLE_OBJC_ARC = YES;
GCC_PRECOMPILE_PREFIX_HEADER = YES;
GCC_PREFIX_HEADER = "Capitals/Capitals-Prefix.pch";
INFOPLIST_FILE = "Capitals/Capitals-Info.plist";
IPHONEOS_DEPLOYMENT_TARGET = 4.3;
IPHONEOS_DEPLOYMENT_TARGET = 5.0;
PRODUCT_NAME = "World Capitals";
WRAPPER_EXTENSION = app;
};

View file

@ -43,14 +43,14 @@
MWMPin * pin = [MWMApi pinFromUrl:url];
if (pin)
{
size_t const cityId = [pin.idOrUrl integerValue];
NSInteger const cityId = [pin.idOrUrl integerValue];
// display selected page based on passed id
if ([[UIDevice currentDevice] userInterfaceIdiom] == UIUserInterfaceIdiomPhone)
if ([UIDevice currentDevice].userInterfaceIdiom == UIUserInterfaceIdiomPhone)
{
[self.navigationController popToRootViewControllerAnimated:NO];
MasterViewController * masterVC = [self.navigationController.viewControllers objectAtIndex:0];
MasterViewController * masterVC = self.navigationController.viewControllers[0];
if (!masterVC.detailViewController)
masterVC.detailViewController = [[[CityDetailViewController alloc] initWithNibName:@"CityDetailViewController" bundle:nil] autorelease];
masterVC.detailViewController = [[CityDetailViewController alloc] initWithNibName:@"CityDetailViewController" bundle:nil];
masterVC.detailViewController.cityIndex = cityId;
[masterVC.navigationController pushViewController:masterVC.detailViewController animated:YES];
}
@ -65,35 +65,27 @@
return NO;
}
- (void)dealloc
{
self.window = nil;
self.navigationController = nil;
self.splitViewController = nil;
[super dealloc];
}
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
{
self.window = [[[UIWindow alloc] initWithFrame:[[UIScreen mainScreen] bounds]] autorelease];
self.window = [[UIWindow alloc] initWithFrame:[UIScreen mainScreen].bounds];
if ([[UIDevice currentDevice] userInterfaceIdiom] == UIUserInterfaceIdiomPhone)
if ([UIDevice currentDevice].userInterfaceIdiom == UIUserInterfaceIdiomPhone)
{
MasterViewController * masterViewController = [[[MasterViewController alloc] initWithNibName:@"MasterViewController" bundle:nil] autorelease];
self.navigationController = [[[UINavigationController alloc] initWithRootViewController:masterViewController] autorelease];
MasterViewController * masterViewController = [[MasterViewController alloc] initWithNibName:@"MasterViewController" bundle:nil];
self.navigationController = [[UINavigationController alloc] initWithRootViewController:masterViewController];
self.window.rootViewController = self.navigationController;
}
else
{
MasterViewController * masterViewController = [[[MasterViewController alloc] initWithNibName:@"MasterViewController" bundle:nil] autorelease];
UINavigationController * masterNavigationController = [[[UINavigationController alloc] initWithRootViewController:masterViewController] autorelease];
MasterViewController * masterViewController = [[MasterViewController alloc] initWithNibName:@"MasterViewController" bundle:nil];
UINavigationController * masterNavigationController = [[UINavigationController alloc] initWithRootViewController:masterViewController];
CityDetailViewController * detailViewController = [[[CityDetailViewController alloc] initWithNibName:@"CityDetailViewController" bundle:nil] autorelease];
UINavigationController * detailNavigationController = [[[UINavigationController alloc] initWithRootViewController:detailViewController] autorelease];
CityDetailViewController * detailViewController = [[CityDetailViewController alloc] initWithNibName:@"CityDetailViewController" bundle:nil];
UINavigationController * detailNavigationController = [[UINavigationController alloc] initWithRootViewController:detailViewController];
masterViewController.detailViewController = detailViewController;
self.splitViewController = [[[UISplitViewController alloc] init] autorelease];
self.splitViewController = [[UISplitViewController alloc] init];
self.splitViewController.delegate = detailViewController;
self.splitViewController.viewControllers = @[masterNavigationController, detailNavigationController];

View file

@ -1,41 +0,0 @@
/*******************************************************************************
Copyright (c) 2013, MapsWithMe GmbH
All rights reserved.
Redistribution and use in source and binary forms, with or without
modification, are permitted provided that the following conditions are met:
* Redistributions of source code must retain the above copyright notice, this
list of conditions and the following disclaimer.
* Redistributions in binary form must reproduce the above copyright notice,
this list of conditions and the following disclaimer in the documentation
and/or other materials provided with the distribution.
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
******************************************************************************/
@class NSString;
typedef struct
{
NSString * name;
double lat;
double lon;
NSString * countryCode;
int population;
NSString * timeZone;
} City;
extern City const CAPITALS[241];

View file

@ -1,272 +0,0 @@
/*******************************************************************************
Copyright (c) 2013, MapsWithMe GmbH
All rights reserved.
Redistribution and use in source and binary forms, with or without
modification, are permitted provided that the following conditions are met:
* Redistributions of source code must retain the above copyright notice, this
list of conditions and the following disclaimer.
* Redistributions in binary form must reproduce the above copyright notice,
this list of conditions and the following disclaimer in the documentation
and/or other materials provided with the distribution.
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
******************************************************************************/
#include "City.h"
City const CAPITALS[241] = {{@"Abu Dhabi", 24.46667, 54.36667, @"AE", 603492, @"Asia/Dubai"},
{@"Abuja", 9.06853, 7.48375, @"NG", 590400, @"Africa/Lagos"},
{@"Accra", 5.55602, -0.1969, @"GH", 1963264, @"Africa/Accra"},
{@"Adamstown", -25.06597, -130.1015, @"PN", 46, @"Pacific/Pitcairn"},
{@"Addis Ababa", 9.02497, 38.74689, @"ET", 2757729, @"Africa/Addis_Ababa"},
{@"Algiers", 36.7525, 3.04197, @"DZ", 1977663, @"Africa/Algiers"},
{@"Alofi", -19.05952, -169.9187, @"NU", 624, @"Pacific/Niue"},
{@"Amman", 31.95522, 35.94503, @"JO", 1275857, @"Asia/Amman"},
{@"Amsterdam", 52.37403, 4.88969, @"NL", 741636, @"Europe/Amsterdam"},
{@"Andorra la Vella", 42.50779, 1.52109, @"AD", 20430, @"Europe/Andorra"},
{@"Ankara", 39.91987, 32.85427, @"TR", 3517182, @"Europe/Istanbul"},
{@"Antananarivo", -18.91368, 47.53613, @"MG", 1391433, @"Indian/Antananarivo"},
{@"Apia", -13.83333, -171.7667, @"WS", 40407, @"Pacific/Apia"},
{@"Ashgabat", 37.95, 58.38333, @"TM", 727700, @"Asia/Ashgabat"},
{@"Asmara", 15.33805, 38.93184, @"ER", 563930, @"Africa/Asmara"},
{@"Astana", 51.1801, 71.44598, @"KZ", 345604, @"Asia/Almaty"},
{@"Asunción", -25.30066, -57.63591, @"PY", 1482200, @"America/Asuncion"},
{@"Athens", 37.97945, 23.71622, @"GR", 729137, @"Europe/Athens"},
{@"Avarua", -21.20778, -159.775, @"CK", 13373, @"Pacific/Rarotonga"},
{@"Baghdad", 33.34058, 44.40088, @"IQ", 5672513, @"Asia/Baghdad"},
{@"Baku", 40.37767, 49.89201, @"AZ", 1116513, @"Asia/Baku"},
{@"Bamako", 12.65, -8, @"ML", 1297281, @"Africa/Bamako"},
{@"Bandar Seri Begawan", 4.94029, 114.9481, @"BN", 64409, @"Asia/Brunei"},
{@"Bangkok", 13.75398, 100.5014, @"TH", 5104476, @"Asia/Bangkok"},
{@"Bangui", 4.36122, 18.55496, @"CF", 542393, @"Africa/Bangui"},
{@"Banjul", 13.45274, -16.57803, @"GM", 34589, @"Africa/Banjul"},
{@"Basse-Terre", 15.99854, -61.72548, @"GP", 11472, @"America/Guadeloupe"},
{@"Basseterre", 17.29484, -62.7261, @"KN", 12920, @"America/St_Kitts"},
{@"Beijing", 39.9075, 116.3972, @"CN", 7480601, @"Asia/Shanghai"},
{@"Beirut", 33.88894, 35.49442, @"LB", 1916100, @"Asia/Beirut"},
{@"Belgrade", 44.80401, 20.46513, @"RS", 1273651, @"Europe/Belgrade"},
{@"Belmopan", 17.25, -88.76667, @"BZ", 13381, @"America/Belize"},
{@"Berlin", 52.52437, 13.41053, @"DE", 3426354, @"Europe/Berlin"},
{@"Bern", 46.94809, 7.44744, @"CH", 121631, @"Europe/Zurich"},
{@"Bishkek", 42.87, 74.59, @"KG", 900000, @"Asia/Bishkek"},
{@"Bissau", 11.86357, -15.59767, @"GW", 388028, @"Africa/Bissau"},
{@"Bogotá", 4.60971, -74.08175, @"CO", 7102602, @"America/Bogota"},
{@"Brasília", -15.77972, -47.92972, @"BR", 2207718, @"America/Sao_Paulo"},
{@"Bratislava", 48.14816, 17.10674, @"SK", 423737, @"Europe/Bratislava"},
{@"Brazzaville", -4.26613, 15.28318, @"CG", 1284609, @"Africa/Brazzaville"},
{@"Bridgetown", 13.1, -59.61667, @"BB", 98511, @"America/Barbados"},
{@"Brussels", 50.85045, 4.34878, @"BE", 1019022, @"Europe/Brussels"},
{@"Bucharest", 44.43225, 26.10626, @"RO", 1877155, @"Europe/Bucharest"},
{@"Budapest", 47.49801, 19.03991, @"HU", 1696128, @"Europe/Budapest"},
{@"Buenos Aires", -34.61315, -58.37723, @"AR", 13076300, @"America/Argentina/Buenos_Aires"},
{@"Bujumbura", -3.3822, 29.3644, @"BI", 331700, @"Africa/Bujumbura"},
{@"Cairo", 30.06263, 31.24967, @"EG", 7734614, @"Africa/Cairo"},
{@"Canberra", -35.28346, 149.1281, @"AU", 327700, @"Australia/Sydney"},
{@"Caracas", 10.48801, -66.87919, @"VE", 3000000, @"America/Caracas"},
{@"Castries", 13.9957, -61.00614, @"LC", 10000, @"America/St_Lucia"},
{@"Cayenne", 4.93333, -52.33333, @"GF", 61550, @"America/Cayenne"},
{@"Charlotte Amalie", 18.3419, -64.9307, @"VI", 20000, @"America/St_Thomas"},
{@"Chişinău", 47.00556, 28.8575, @"MD", 635994, @"Europe/Chisinau"},
{@"Cockburn Town", 21.46122, -71.14188, @"TC", 3720, @"America/Grand_Turk"},
{@"Colombo", 6.93194, 79.84778, @"LK", 648034, @"Asia/Colombo"},
{@"Conakry", 9.53795, -13.67729, @"GN", 1767200, @"Africa/Conakry"},
{@"Copenhagen", 55.67594, 12.56553, @"DK", 1153615, @"Europe/Copenhagen"},
{@"Dakar", 14.6937, -17.44406, @"SN", 2476400, @"Africa/Dakar"},
{@"Damascus", 33.5102, 36.29128, @"SY", 1569394, @"Asia/Damascus"},
{@"Dhaka", 23.7104, 90.40744, @"BD", 10356500, @"Asia/Dhaka"},
{@"Dili", -8.55861, 125.5736, @"TL", 150000, @"Asia/Dili"},
{@"Djibouti", 11.58901, 43.14503, @"DJ", 623891, @"Africa/Djibouti"},
{@"Dodoma", -6.17221, 35.73947, @"TZ", 180541, @"Africa/Dar_es_Salaam"},
{@"Doha", 25.27932, 51.52245, @"QA", 344939, @"Asia/Qatar"},
{@"Douglas", 54.15, -4.48333, @"IM", 26218, @"Europe/Isle_of_Man"},
{@"Dublin", 53.33306, -6.24889, @"IE", 1024027, @"Europe/Dublin"},
{@"Dushanbe", 38.53575, 68.77905, @"TJ", 543107, @"Asia/Dushanbe"},
{@"Flying Fish Cove", -10.42172, 105.6791, @"CX", 500, @"Indian/Christmas"},
{@"Fort-de-France", 14.60892, -61.07334, @"MQ", 89995, @"America/Martinique"},
{@"Freetown", 8.484, -13.22994, @"SL", 802639, @"Africa/Freetown"},
{@"Funafuti", -8.52425, 179.1942, @"TV", 4492, @"Pacific/Funafuti"},
{@"Gaborone", -24.65451, 25.90859, @"BW", 208411, @"Africa/Gaborone"},
{@"George Town", 19.28692, -81.36706, @"KY", 29370, @"America/Cayman"},
{@"Georgetown", 6.80448, -58.15527, @"GY", 235017, @"America/Guyana"},
{@"Gibraltar", 36.14474, -5.35257, @"GI", 26544, @"Europe/Gibraltar"},
{@"Grytviken", -54.28111, -36.5092, @"GS", 2, @"Atlantic/South_Georgia"},
{@"Guatemala City", 14.64072, -90.51327, @"GT", 994938, @"America/Guatemala"},
{@"Gustavia", 17.89618, -62.84978, @"BL", 5988, @"America/St_Barthelemy"},
{@"Hagåtña", 13.47567, 144.7489, @"GU", 1051, @"Pacific/Guam"},
{@"Hamilton", 32.29149, -64.77797, @"BM", 902, @"Atlantic/Bermuda"},
{@"Harare", -17.82772, 31.05337, @"ZW", 1542813, @"Africa/Harare"},
{@"Havana", 23.13302, -82.38304, @"CU", 2163824, @"America/Havana"},
{@"Helsinki", 60.16952, 24.93545, @"FI", 558457, @"Europe/Helsinki"},
{@"Hong Kong", 22.28552, 114.1577, @"HK", 7012738, @"Asia/Hong_Kong"},
{@"Honiara", -9.43333, 159.95, @"SB", 56298, @"Pacific/Guadalcanal"},
{@"Hà Nội", 21.0245, 105.8412, @"VN", 1431270, @"Asia/Ho_Chi_Minh"},
{@"Islamabad", 33.72148, 73.04329, @"PK", 601600, @"Asia/Karachi"},
{@"Jakarta", -6.21462, 106.8451, @"ID", 8540121, @"Asia/Jakarta"},
{@"Jamestown", -15.93872, -5.71675, @"SH", 637, @"Atlantic/St_Helena"},
{@"Juba", 4.85165, 31.58247, @"SS", 300000, @"Africa/Juba"},
{@"Kabul", 34.52813, 69.17233, @"AF", 3043532, @"Asia/Kabul"},
{@"Kampala", 0.31628, 32.58219, @"UG", 1353189, @"Africa/Kampala"},
{@"Kathmandu", 27.70169, 85.3206, @"NP", 1442271, @"Asia/Kathmandu"},
{@"Khartoum", 15.55177, 32.53241, @"SD", 1974647, @"Africa/Khartoum"},
{@"Kiev", 50.45466, 30.5238, @"UA", 2514227, @"Europe/Kiev"},
{@"Kigali", -1.94995, 30.05885, @"RW", 745261, @"Africa/Kigali"},
{@"Kingston", -29.05459, 167.9663, @"NF", 880, @"Pacific/Norfolk"},
{@"Kingston", 17.99702, -76.79358, @"JM", 937700, @"America/Jamaica"},
{@"Kingstown", 13.15872, -61.22475, @"VC", 24518, @"America/St_Vincent"},
{@"Kinshasa", -4.32142, 15.30807, @"CD", 7785965, @"Africa/Kinshasa"},
{@"Kralendijk", 12.15, -68.26667, @"BQ", 3081, @"America/Kralendijk"},
{@"Kuala Lumpur", 3.1412, 101.6865, @"MY", 1453975, @"Asia/Kuala_Lumpur"},
{@"Kuwait City", 29.36972, 47.97833, @"KW", 60064, @"Asia/Kuwait"},
{@"El Aaiún", 27.16224, -13.20315, @"EH", 188084, @"Africa/El_Aaiun"},
{@"Libreville", 0.39241, 9.45356, @"GA", 578156, @"Africa/Libreville"},
{@"Lilongwe", -13.96692, 33.78725, @"MW", 646750, @"Africa/Blantyre"},
{@"Lima", -12.04318, -77.02824, @"PE", 7737002, @"America/Lima"},
{@"Lisbon", 38.71667, -9.13333, @"PT", 517802, @"Europe/Lisbon"},
{@"Ljubljana", 46.05108, 14.50513, @"SI", 255115, @"Europe/Ljubljana"},
{@"Lomé", 6.13748, 1.21227, @"TG", 749700, @"Africa/Lome"},
{@"London", 51.50853, -0.12574, @"GB", 7556900, @"Europe/London"},
{@"Longyearbyen", 78.2186, 15.64007, @"SJ", 2060, @"Arctic/Longyearbyen"},
{@"Luanda", -8.83682, 13.23432, @"AO", 2776168, @"Africa/Luanda"},
{@"Lusaka", -15.40669, 28.28713, @"ZM", 1267440, @"Africa/Lusaka"},
{@"Luxembourg", 49.61167, 6.13, @"LU", 76684, @"Europe/Luxembourg"},
{@"Macau", 22.20056, 113.5461, @"MO", 520400, @"Asia/Macau"},
{@"Madrid", 40.4165, -3.70256, @"ES", 3255944, @"Europe/Madrid"},
{@"Majuro", 7.08971, 171.3803, @"MH", 25400, @"Pacific/Majuro"},
{@"Malabo", 3.75, 8.78333, @"GQ", 155963, @"Africa/Malabo"},
{@"Male", 4.1748, 73.50888, @"MV", 103693, @"Indian/Maldives"},
{@"Mamoudzou", -12.77944, 45.22722, @"YT", 54831, @"Indian/Mayotte"},
{@"Managua", 12.13282, -86.2504, @"NI", 973087, @"America/Managua"},
{@"Manama", 26.21536, 50.5832, @"BH", 147074, @"Asia/Bahrain"},
{@"Manila", 14.6042, 120.9822, @"PH", 10444527, @"Asia/Manila"},
{@"Maputo", -25.96553, 32.58322, @"MZ", 1191613, @"Africa/Maputo"},
{@"Mariehamn", 60.09726, 19.93481, @"AX", 10682, @"Europe/Mariehamn"},
{@"Marigot", 18.06667, -63.08333, @"MF", 5700, @"America/Marigot"},
{@"Maseru", -29.31667, 27.48333, @"LS", 118355, @"Africa/Maseru"},
{@"Mata-Utu", -13.28163, -176.1745, @"WF", 1200, @"Pacific/Wallis"},
{@"Mbabane", -26.31667, 31.13333, @"SZ", 76218, @"Africa/Mbabane"},
{@"Melekeok", 7.50043, 134.6235, @"PW", 0, @"Pacific/Palau"},
{@"Mexico City", 19.42847, -99.12766, @"MX", 12294193, @"America/Mexico_City"},
{@"Minsk", 53.9, 27.56667, @"BY", 1742124, @"Europe/Minsk"},
{@"Mogadishu", 2.03711, 45.34375, @"SO", 2587183, @"Africa/Mogadishu"},
{@"Monaco", 43.73333, 7.41667, @"MC", 32965, @"Europe/Monaco"},
{@"Monrovia", 6.30054, -10.7969, @"LR", 939524, @"Africa/Monrovia"},
{@"Montevideo", -34.83346, -56.16735, @"UY", 1270737, @"America/Montevideo"},
{@"Moroni", -11.70216, 43.25506, @"KM", 42872, @"Indian/Comoro"},
{@"Moscow", 55.75222, 37.61556, @"RU", 10381222, @"Europe/Moscow"},
{@"Muscat", 23.61387, 58.5922, @"OM", 797000, @"Asia/Muscat"},
{@"N'Djamena", 12.10672, 15.0444, @"TD", 721081, @"Africa/Ndjamena"},
{@"Nairobi", -1.28333, 36.81667, @"KE", 2750547, @"Africa/Nairobi"},
{@"Nassau", 25.05823, -77.34306, @"BS", 227940, @"America/Nassau"},
{@"Nay Pyi Taw", 19.745, 96.12972, @"MM", 925000, @"Asia/Rangoon"},
{@"New Delhi", 28.63576, 77.22445, @"IN", 317797, @"Asia/Kolkata"},
{@"Niamey", 13.51366, 2.1098, @"NE", 774235, @"Africa/Niamey"},
{@"Nicosia", 35.16667, 33.36667, @"CY", 200452, @"Asia/Nicosia"},
{@"Nouakchott", 18.08581, -15.9785, @"MR", 661400, @"Africa/Nouakchott"},
{@"Nouméa", -22.27631, 166.4572, @"NC", 93060, @"Pacific/Noumea"},
{@"Nukualofa", -21.13938, -175.2018, @"TO", 22400, @"Pacific/Tongatapu"},
{@"Nuuk", 64.18347, -51.72157, @"GL", 14798, @"America/Godthab"},
{@"Oranjestad", 12.52398, -70.02703, @"AW", 29998, @"America/Aruba"},
{@"Oslo", 59.91273, 10.74609, @"NO", 580000, @"Europe/Oslo"},
{@"Ottawa", 45.41117, -75.69812, @"CA", 812129, @"America/Toronto"},
{@"Ouagadougou", 12.36566, -1.53388, @"BF", 1086505, @"Africa/Ouagadougou"},
{@"Pago Pago", -14.27806, -170.7025, @"AS", 11500, @"Pacific/Pago_Pago"},
{@"Palikir - National Government Center", 6.92477, 158.1611, @"FM", 0, @"Pacific/Pohnpei"},
{@"Panamá", 8.9936, -79.51973, @"PA", 408168, @"America/Panama"},
{@"Papeete", -17.53333, -149.5667, @"PF", 26357, @"Pacific/Tahiti"},
{@"Paramaribo", 5.86638, -55.16682, @"SR", 223757, @"America/Paramaribo"},
{@"Paris", 48.85341, 2.3488, @"FR", 2138551, @"Europe/Paris"},
{@"Philipsburg", 18.026, -63.04582, @"SX", 1400, @"America/Lower_Princes"},
{@"Phnom Penh", 11.56245, 104.916, @"KH", 1573544, @"Asia/Phnom_Penh"},
{@"Plymouth", 16.70555, -62.21292, @"MS", 0, @"America/Montserrat"},
{@"Podgorica", 42.44111, 19.26361, @"ME", 136473, @"Europe/Podgorica"},
{@"Port Louis", -20.16194, 57.49889, @"MU", 155226, @"Indian/Mauritius"},
{@"Port Moresby", -9.44314, 147.1797, @"PG", 283733, @"Pacific/Port_Moresby"},
{@"Port-Vila", -17.73381, 168.3219, @"VU", 35901, @"Pacific/Efate"},
{@"Port-au-Prince", 18.53917, -72.335, @"HT", 1234742, @"America/Port-au-Prince"},
{@"Port-aux-Français", -49.35, 70.21667, @"TF", 45, @"Indian/Kerguelen"},
{@"Port-of-Spain", 10.66617, -61.51657, @"TT", 49031, @"America/Port_of_Spain"},
{@"Porto-Novo", 6.49646, 2.60359, @"BJ", 234168, @"Africa/Porto-Novo"},
{@"Prague", 50.08804, 14.42076, @"CZ", 1165581, @"Europe/Prague"},
{@"Praia", 14.93152, -23.51254, @"CV", 113364, @"Atlantic/Cape_Verde"},
{@"Pretoria", -25.74486, 28.18783, @"ZA", 1619438, @"Africa/Johannesburg"},
{@"Pristina", 42.67272, 21.16688, @"XK", 550000, @"Europe/Belgrade"},
{@"Pyongyang", 39.03385, 125.7543, @"KP", 3222000, @"Asia/Pyongyang"},
{@"Quito", -0.22985, -78.52495, @"EC", 1399814, @"America/Guayaquil"},
{@"Rabat", 34.01325, -6.83255, @"MA", 1655753, @"Africa/Casablanca"},
{@"Reykjavík", 64.13548, -21.89541, @"IS", 113906, @"Atlantic/Reykjavik"},
{@"Riga", 56.946, 24.10589, @"LV", 742572, @"Europe/Riga"},
{@"Riyadh", 24.68773, 46.72185, @"SA", 4205961, @"Asia/Riyadh"},
{@"Road Town", 18.41667, -64.61667, @"VG", 8449, @"America/Tortola"},
{@"Rome", 41.89474, 12.4839, @"IT", 2563241, @"Europe/Rome"},
{@"Roseau", 15.30174, -61.38808, @"DM", 16571, @"America/Dominica"},
{@"Saint George's", 12.05644, -61.74849, @"GD", 7500, @"America/Grenada"},
{@"Saint Helier", 49.18804, -2.10491, @"JE", 28000, @"Europe/Jersey"},
{@"Saint Johns", 17.11667, -61.85, @"AG", 24226, @"America/Antigua"},
{@"Saint Peter Port", 49.45981, -2.53527, @"GG", 16488, @"Europe/Guernsey"},
{@"Saint-Denis", -20.88231, 55.4504, @"RE", 137195, @"Indian/Reunion"},
{@"Saint-Pierre", 46.78091, -56.17196, @"PM", 6200, @"America/Miquelon"},
{@"Saipan", 15.21233, 145.7545, @"MP", 48220, @"Pacific/Saipan"},
{@"San José", 9.93333, -84.08333, @"CR", 335007, @"America/Costa_Rica"},
{@"San Juan", 18.46633, -66.10572, @"PR", 418140, @"America/Puerto_Rico"},
{@"San Marino", 43.93667, 12.44639, @"SM", 4500, @"Europe/San_Marino"},
{@"San Salvador", 13.68935, -89.18718, @"SV", 525990, @"America/El_Salvador"},
{@"Sanaa", 15.35472, 44.20667, @"YE", 1937451, @"Asia/Aden"},
{@"Santiago", -33.45694, -70.64827, @"CL", 4837295, @"America/Santiago"},
{@"Santo Domingo", 18.50012, -69.98857, @"DO", 2201941, @"America/Santo_Domingo"},
{@"Sarajevo", 43.84864, 18.35644, @"BA", 696731, @"Europe/Sarajevo"},
{@"Seoul", 37.56826, 126.9778, @"KR", 10349312, @"Asia/Seoul"},
{@"Singapore", 1.28967, 103.8501, @"SG", 3547809, @"Asia/Singapore"},
{@"Skopje", 42.00122, 21.42878, @"MK", 474889, @"Europe/Skopje"},
{@"Sofia", 42.69751, 23.32415, @"BG", 1152556, @"Europe/Sofia"},
{@"Stanley", -51.7, -57.85, @"FK", 2213, @"Atlantic/Stanley"},
{@"Stockholm", 59.33258, 18.0649, @"SE", 1253309, @"Europe/Stockholm"},
{@"Sucre", -19.03332, -65.26274, @"BO", 224838, @"America/La_Paz"},
{@"Suva", -18.14161, 178.4415, @"FJ", 77366, @"Pacific/Fiji"},
{@"São Tomé", 0.33654, 6.72732, @"ST", 53300, @"Africa/Sao_Tome"},
{@"Taipei", 25.04776, 121.5319, @"TW", 7871900, @"Asia/Taipei"},
{@"Tallinn", 59.43696, 24.75353, @"EE", 394024, @"Europe/Tallinn"},
{@"Tarawa", 1.3278, 172.977, @"KI", 40311, @"Pacific/Tarawa"},
{@"Tashkent", 41.26465, 69.21627, @"UZ", 1978028, @"Asia/Tashkent"},
{@"Tbilisi", 41.69411, 44.83368, @"GE", 1049498, @"Asia/Tbilisi"},
{@"Tegucigalpa", 14.0818, -87.20681, @"HN", 850848, @"America/Tegucigalpa"},
{@"Tehrān", 35.69439, 51.42151, @"IR", 7153309, @"Asia/Tehran"},
{@"The Valley", 18.21704, -63.05783, @"AI", 2035, @"America/Anguilla"},
{@"Thimphu", 27.46609, 89.64191, @"BT", 98676, @"Asia/Thimphu"},
{@"Tirana", 41.3275, 19.81889, @"AL", 374801, @"Europe/Tirane"},
{@"Tokyo", 35.6895, 139.6917, @"JP", 8336599, @"Asia/Tokyo"},
{@"Tripoli", 32.87519, 13.18746, @"LY", 1150989, @"Africa/Tripoli"},
{@"Tunis", 36.81897, 10.16579, @"TN", 693210, @"Africa/Tunis"},
{@"Tórshavn", 62.00973, -6.77164, @"FO", 13200, @"Atlantic/Faroe"},
{@"Ulaanbaatar", 47.90771, 106.8832, @"MN", 844818, @"Asia/Ulaanbaatar"},
{@"Vaduz", 47.14151, 9.52154, @"LI", 5197, @"Europe/Vaduz"},
{@"Valletta", 35.89972, 14.51472, @"MT", 6794, @"Europe/Malta"},
{@"Vatican City", 41.90236, 12.45332, @"VA", 829, @"Europe/Vatican"},
{@"Victoria", -4.61667, 55.45, @"SC", 22881, @"Indian/Mahe"},
{@"Vienna", 48.20849, 16.37208, @"AT", 1691468, @"Europe/Vienna"},
{@"Vientiane", 17.96667, 102.6, @"LA", 196731, @"Asia/Vientiane"},
{@"Vilnius", 54.68916, 25.2798, @"LT", 542366, @"Europe/Vilnius"},
{@"Warsaw", 52.22977, 21.01178, @"PL", 1702139, @"Europe/Warsaw"},
{@"Washington, D. C.", 38.89511, -77.03637, @"US", 601723, @"America/New_York"},
{@"Wellington", -41.28664, 174.7756, @"NZ", 381900, @"Pacific/Auckland"},
{@"West Island", -12.15681, 96.82251, @"CC", 120, @"Indian/Cocos"},
{@"Willemstad", 12.1084, -68.93354, @"CW", 125000, @"America/Curacao"},
{@"Windhoek", -22.55941, 17.08323, @"NA", 268132, @"Africa/Windhoek"},
{@"Yamoussoukro", 6.82055, -5.27674, @"CI", 194530, @"Africa/Abidjan"},
{@"Yaoundé", 3.86667, 11.51667, @"CM", 1299369, @"Africa/Douala"},
{@"Yerevan", 40.18111, 44.51361, @"AM", 1093485, @"Asia/Yerevan"},
{@"Zagreb", 45.81444, 15.97798, @"HR", 698966, @"Europe/Zagreb"}
};

View file

@ -30,6 +30,8 @@
@interface CityDetailViewController : UITableViewController <UISplitViewControllerDelegate>
@property (nonatomic, assign) size_t cityIndex;
@property (nonatomic, assign) NSInteger cityIndex;
@property (strong, nonatomic) NSDictionary * city;
@property (strong, nonatomic) UIPopoverController * masterPopoverController;
@end

View file

@ -27,13 +27,13 @@
******************************************************************************/
#import "CityDetailViewController.h"
#import "City.h"
#import "MapsWithMeAPI.h"
@interface CityDetailViewController ()
@property (strong, nonatomic) UIPopoverController * masterPopoverController;
- (void)configureView;
@end
@ -41,27 +41,20 @@
- (NSString *)urlEncode:(NSString *)str
{
return [(NSString *)CFURLCreateStringByAddingPercentEscapes(kCFAllocatorDefault, (CFStringRef)str, NULL, CFSTR("!$&'()*+,-./:;=?@_~"), kCFStringEncodingUTF8) autorelease];
return (__bridge_transfer NSString *)CFURLCreateStringByAddingPercentEscapes(kCFAllocatorDefault, (CFStringRef)str, NULL, CFSTR("!$&'()*+,-./:;=?@_~"), kCFStringEncodingUTF8);
}
- (void)showCapitalOnTheMap:(BOOL)withLink
{
City const * city = &CAPITALS[_cityIndex];
NSString * pinId;
if (withLink)
pinId = [NSString stringWithFormat:@"http://en.wikipedia.org/wiki/%@", [self urlEncode:city->name]];
pinId = [NSString stringWithFormat:@"http://en.wikipedia.org/wiki/%@", [self urlEncode:self.city[@"name"]]];
else
pinId = [NSString stringWithFormat:@"%ld", _cityIndex];
[MWMApi showLat:city->lat lon:city->lon title:city->name andId:pinId];
pinId = [NSString stringWithFormat:@"%i", _cityIndex];
[MWMApi showLat:[self.city[@"lat"] doubleValue] lon:[self.city[@"lon"] doubleValue] title:self.city[@"name"] andId:pinId];
}
- (void)dealloc
{
self.masterPopoverController = nil;
[super dealloc];
}
- (void)setCityIndex:(size_t)newCityIndex
- (void)setCityIndex:(NSInteger)newCityIndex
{
if (_cityIndex != newCityIndex)
{
@ -70,13 +63,13 @@
[self configureView];
}
if (self.masterPopoverController != nil)
if (self.masterPopoverController)
[self.masterPopoverController dismissPopoverAnimated:YES];
}
- (void)configureView
{
self.title = CAPITALS[_cityIndex].name;
self.title = self.city[@"name"];
[self.tableView reloadData];
}
@ -106,7 +99,7 @@
{
if (indexPath.section == 0)
{
cell = [[[UITableViewCell alloc] initWithStyle:UITableViewCellStyleValue1 reuseIdentifier:cellId] autorelease];
cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleValue1 reuseIdentifier:cellId];
cell.selectionStyle = UITableViewCellSelectionStyleNone;
switch (indexPath.row)
{
@ -120,7 +113,7 @@
}
else
{
cell = [[[UITableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:cellId] autorelease];
cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:cellId];
cell.textLabel.textAlignment = UITextAlignmentCenter;
if (indexPath.section == 1)
cell.textLabel.text = @"Show map and come back";
@ -131,14 +124,13 @@
if (indexPath.section == 0)
{
City const * city = &CAPITALS[_cityIndex];
switch (indexPath.row)
{
case 0: cell.detailTextLabel.text = [NSString stringWithFormat:@"%lf", city->lat]; break;
case 1: cell.detailTextLabel.text = [NSString stringWithFormat:@"%lf", city->lon]; break;
case 2: cell.detailTextLabel.text = city->countryCode; break;
case 3: cell.detailTextLabel.text = [NSString stringWithFormat:@"%d", city->population]; break;
case 4: cell.detailTextLabel.text = city->timeZone; break;
case 0: cell.detailTextLabel.text = [self.city[@"lat"] stringValue]; break;
case 1: cell.detailTextLabel.text = [self.city[@"lon"] stringValue]; break;
case 2: cell.detailTextLabel.text = self.city[@"countryCode"]; break;
case 3: cell.detailTextLabel.text = [self.city[@"population"] stringValue]; break;
case 4: cell.detailTextLabel.text = self.city[@"timeZone"]; break;
default: break;
}
}

View file

@ -28,29 +28,51 @@
#import "MasterViewController.h"
#import "CityDetailViewController.h"
#import "City.h"
#import "MapsWithMeAPI.h"
@interface MasterViewController ()
@property (strong, nonatomic) NSArray * capitals;
@end
@implementation MasterViewController
- (CityDetailViewController *)detailViewController
{
if (!_detailViewController)
_detailViewController = [[CityDetailViewController alloc] initWithNibName:@"CityDetailViewController" bundle:nil];
return _detailViewController;
}
- (NSArray *)capitals
{
if (!_capitals)
{
NSString * path = [[NSBundle mainBundle] pathForResource:@"capitals" ofType:@"plist"];
_capitals = [NSArray arrayWithContentsOfFile:path];
}
return _capitals;
}
- (void)showAllCitiesOnTheMap:(id)sender
{
size_t const capitalsCount = sizeof(CAPITALS)/sizeof(CAPITALS[0]);
NSMutableArray * array = [[NSMutableArray alloc] initWithCapacity:[self.capitals count]];
NSMutableArray * array = [[NSMutableArray alloc] initWithCapacity:capitalsCount];
for (size_t i = 0; i < capitalsCount; ++i)
for (NSInteger i = 0; i < [self.capitals count]; ++i)
{
NSString * pinId = [[[NSString alloc] initWithFormat:@"%ld", i] autorelease];
NSString * pinId = [[NSString alloc] initWithFormat:@"%i", i];
// Note that url is empty - it means "More details" button for a pin in MapsWithMe will lead back to this example app
MWMPin * pin = [[[MWMPin alloc] initWithLat:CAPITALS[i].lat lon:CAPITALS[i].lon title:CAPITALS[i].name andId:pinId] autorelease];
NSDictionary * city = self.capitals[i];
MWMPin * pin = [[MWMPin alloc] initWithLat:[city[@"lat"] doubleValue] lon:[city[@"lon"] doubleValue] title:city[@"name"] andId:pinId];
[array addObject:pin];
}
// Your should hide any top view objects like UIPopoverController before calling +showPins:
// If user does not installed MapsWithMe app, a popup dialog will be shown
[self.detailViewController.masterPopoverController dismissPopoverAnimated:YES];
[MWMApi showPins:array];
[array release];
}
- (id)initWithNibName:(NSString *)nibNameOrNil bundle:(NSBundle *)nibBundleOrNil
@ -59,7 +81,7 @@
if (self)
{
self.title = NSLocalizedString(@"World Capitals", nil);
if ([[UIDevice currentDevice] userInterfaceIdiom] == UIUserInterfaceIdiomPad)
if ([UIDevice currentDevice].userInterfaceIdiom == UIUserInterfaceIdiomPad)
{
self.clearsSelectionOnViewWillAppear = NO;
self.contentSizeForViewInPopover = CGSizeMake(320.0, 600.0);
@ -68,17 +90,11 @@
return self;
}
- (void)dealloc
{
self.detailViewController = nil;
[super dealloc];
}
- (void)viewDidLoad
{
[super viewDidLoad];
UIBarButtonItem * showMapButton = [[[UIBarButtonItem alloc] initWithTitle:@"Show All" style:UIBarButtonItemStyleDone target:self action:@selector(showAllCitiesOnTheMap:)] autorelease];
UIBarButtonItem * showMapButton = [[UIBarButtonItem alloc] initWithTitle:@"Show All" style:UIBarButtonItemStyleDone target:self action:@selector(showAllCitiesOnTheMap:)];
self.navigationItem.rightBarButtonItem = showMapButton;
}
@ -91,12 +107,12 @@
- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section
{
return sizeof(CAPITALS)/sizeof(CAPITALS[0]);
return [self.capitals count];
}
- (UIView *)tableView:(UITableView *)tableView viewForHeaderInSection:(NSInteger)section
{
UILabel * label = [[[UILabel alloc] initWithFrame:CGRectMake(0, 0, 240, tableView.rowHeight)] autorelease];
UILabel * label = [[UILabel alloc] initWithFrame:CGRectMake(0, 0, 240, tableView.rowHeight)];
label.text = [MWMApi isApiSupported] ? @"MapsWithMe is installed" : @"MapsWithMe is not installed";
label.textAlignment = UITextAlignmentCenter;
label.backgroundColor = [UIColor clearColor];
@ -115,26 +131,21 @@
UITableViewCell * cell = [tableView dequeueReusableCellWithIdentifier:cellId];
if (cell == nil)
{
cell = [[[UITableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:cellId] autorelease];
if ([[UIDevice currentDevice] userInterfaceIdiom] == UIUserInterfaceIdiomPhone)
cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:cellId];
if ([UIDevice currentDevice].userInterfaceIdiom == UIUserInterfaceIdiomPhone)
cell.accessoryType = UITableViewCellAccessoryDisclosureIndicator;
}
cell.textLabel.text = CAPITALS[indexPath.row].name;
cell.textLabel.text = self.capitals[indexPath.row][@"name"];
return cell;
}
- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath
{
if ([[UIDevice currentDevice] userInterfaceIdiom] == UIUserInterfaceIdiomPad)
self.detailViewController.cityIndex = indexPath.row;
else
{
if (!self.detailViewController)
self.detailViewController = [[[CityDetailViewController alloc] initWithNibName:@"CityDetailViewController" bundle:nil] autorelease];
self.detailViewController.cityIndex = indexPath.row;
self.detailViewController.city = self.capitals[indexPath.row];
self.detailViewController.cityIndex = indexPath.row;
if ([UIDevice currentDevice].userInterfaceIdiom == UIUserInterfaceIdiomPhone)
[self.navigationController pushViewController:self.detailViewController animated:YES];
}
}
@end

File diff suppressed because it is too large Load diff