NewsBlur/clients/ios/Other Sources/InAppSettingsKit/Models/IASKSettingsReader.m

272 lines
10 KiB
Mathematica
Raw Normal View History

2013-06-21 17:48:06 -07:00
//
// IASKSettingsReader.m
// http://www.inappsettingskit.com
//
// Copyright (c) 2009:
// Luc Vandal, Edovia Inc., http://www.edovia.com
// Ortwin Gentz, FutureTap GmbH, http://www.futuretap.com
// All rights reserved.
//
// It is appreciated but not required that you give credit to Luc Vandal and Ortwin Gentz,
// as the original authors of this code. You can give credit in a blog post, a tweet or on
// a info page of your app. Also, the original authors appreciate letting them know if you use this code.
//
// This code is licensed under the BSD license that is available at: http://www.opensource.org/licenses/bsd-license.php
//
#import "IASKSettingsReader.h"
#import "IASKSpecifier.h"
#pragma mark -
@interface IASKSettingsReader () {
}
@end
@implementation IASKSettingsReader
- (id) initWithSettingsFileNamed:(NSString*) fileName
applicationBundle:(NSBundle*) bundle {
self = [super init];
if (self) {
_applicationBundle = bundle;
2013-06-21 17:48:06 -07:00
NSString* plistFilePath = [self locateSettingsFile: fileName];
_settingsDictionary = [NSDictionary dictionaryWithContentsOfFile:plistFilePath];
2013-06-21 17:48:06 -07:00
//store the bundle which we'll need later for getting localizations
NSString* settingsBundlePath = [plistFilePath stringByDeletingLastPathComponent];
_settingsBundle = [NSBundle bundleWithPath:settingsBundlePath];
2013-06-21 17:48:06 -07:00
// Look for localization file
self.localizationTable = [_settingsDictionary objectForKey:@"StringsTable"];
if (!self.localizationTable)
{
// Look for localization file using filename
self.localizationTable = [[[[plistFilePath stringByDeletingPathExtension] // removes '.plist'
stringByDeletingPathExtension] // removes potential '.inApp'
lastPathComponent] // strip absolute path
stringByReplacingOccurrencesOfString:[self platformSuffixForInterfaceIdiom:UI_USER_INTERFACE_IDIOM()] withString:@""]; // removes potential '~device' (~ipad, ~iphone)
if([self.settingsBundle pathForResource:self.localizationTable ofType:@"strings"] == nil){
// Could not find the specified localization: use default
self.localizationTable = @"Root";
}
}
if (self.settingsDictionary) {
[self _reinterpretBundle:self.settingsDictionary];
}
}
return self;
}
- (id)initWithFile:(NSString*)file {
return [self initWithSettingsFileNamed:file applicationBundle:[NSBundle mainBundle]];
}
- (id)init {
return [self initWithFile:@"Root"];
}
- (void)setHiddenKeys:(NSSet *)anHiddenKeys {
if (_hiddenKeys != anHiddenKeys) {
_hiddenKeys = anHiddenKeys;
2013-06-21 17:48:06 -07:00
if (self.settingsDictionary) {
[self _reinterpretBundle:self.settingsDictionary];
}
}
}
- (void)_reinterpretBundle:(NSDictionary*)settingsBundle {
NSArray *preferenceSpecifiers = [settingsBundle objectForKey:kIASKPreferenceSpecifiers];
NSInteger sectionCount = -1;
NSMutableArray *dataSource = [NSMutableArray new];
2013-06-21 17:48:06 -07:00
for (NSDictionary *specifier in preferenceSpecifiers) {
if ([self.hiddenKeys containsObject:[specifier objectForKey:kIASKKey]]) {
continue;
}
if ([(NSString*)[specifier objectForKey:kIASKType] isEqualToString:kIASKPSGroupSpecifier]) {
NSMutableArray *newArray = [[NSMutableArray alloc] init];
[newArray addObject:specifier];
[dataSource addObject:newArray];
sectionCount++;
}
else {
if (sectionCount == -1) {
NSMutableArray *newArray = [[NSMutableArray alloc] init];
[dataSource addObject:newArray];
sectionCount++;
}
IASKSpecifier *newSpecifier = [[IASKSpecifier alloc] initWithSpecifier:specifier];
[(NSMutableArray*)[dataSource objectAtIndex:sectionCount] addObject:newSpecifier];
}
}
[self setDataSource:dataSource];
}
- (BOOL)_sectionHasHeading:(NSInteger)section {
return [[[[self dataSource] objectAtIndex:section] objectAtIndex:0] isKindOfClass:[NSDictionary class]];
}
- (NSInteger)numberOfSections {
return [[self dataSource] count];
}
- (NSInteger)numberOfRowsForSection:(NSInteger)section {
int headingCorrection = [self _sectionHasHeading:section] ? 1 : 0;
return [(NSArray*)[[self dataSource] objectAtIndex:section] count] - headingCorrection;
}
- (IASKSpecifier*)specifierForIndexPath:(NSIndexPath*)indexPath {
int headingCorrection = [self _sectionHasHeading:indexPath.section] ? 1 : 0;
IASKSpecifier *specifier = [[[self dataSource] objectAtIndex:indexPath.section] objectAtIndex:(indexPath.row+headingCorrection)];
specifier.settingsReader = self;
return specifier;
}
- (NSIndexPath*)indexPathForKey:(NSString *)key {
for (NSUInteger sectionIndex = 0; sectionIndex < self.dataSource.count; sectionIndex++) {
NSArray *section = [self.dataSource objectAtIndex:sectionIndex];
for (NSUInteger rowIndex = 0; rowIndex < section.count; rowIndex++) {
IASKSpecifier *specifier = (IASKSpecifier*)[section objectAtIndex:rowIndex];
if ([specifier isKindOfClass:[IASKSpecifier class]] && [specifier.key isEqualToString:key]) {
NSUInteger correctedRowIndex = rowIndex - [self _sectionHasHeading:sectionIndex];
return [NSIndexPath indexPathForRow:correctedRowIndex inSection:sectionIndex];
}
}
}
return nil;
}
- (IASKSpecifier*)specifierForKey:(NSString*)key {
for (NSArray *specifiers in _dataSource) {
for (id sp in specifiers) {
if ([sp isKindOfClass:[IASKSpecifier class]]) {
if ([[sp key] isEqualToString:key]) {
return sp;
}
}
}
}
return nil;
}
- (NSString*)titleForSection:(NSInteger)section {
if ([self _sectionHasHeading:section]) {
NSDictionary *dict = [[[self dataSource] objectAtIndex:section] objectAtIndex:kIASKSectionHeaderIndex];
return [self titleForStringId:[dict objectForKey:kIASKTitle]];
}
return nil;
}
- (NSString*)keyForSection:(NSInteger)section {
if ([self _sectionHasHeading:section]) {
return [[[[self dataSource] objectAtIndex:section] objectAtIndex:kIASKSectionHeaderIndex] objectForKey:kIASKKey];
}
return nil;
}
- (NSString*)footerTextForSection:(NSInteger)section {
if ([self _sectionHasHeading:section]) {
NSDictionary *dict = [[[self dataSource] objectAtIndex:section] objectAtIndex:kIASKSectionHeaderIndex];
return [self titleForStringId:[dict objectForKey:kIASKFooterText]];
}
return nil;
}
- (NSString*)titleForStringId:(NSString*)stringId {
return [self.settingsBundle localizedStringForKey:stringId value:stringId table:self.localizationTable];
}
- (NSString*)pathForImageNamed:(NSString*)image {
return [[self.settingsBundle bundlePath] stringByAppendingPathComponent:image];
}
- (NSString *)platformSuffixForInterfaceIdiom:(UIUserInterfaceIdiom) interfaceIdiom {
switch (interfaceIdiom) {
case UIUserInterfaceIdiomPad: return @"~ipad";
case UIUserInterfaceIdiomPhone: return @"~iphone";
2014-09-18 11:25:51 -07:00
default: return @"~iphone";
2013-06-21 17:48:06 -07:00
}
}
- (NSString *)file:(NSString *)file
withBundle:(NSString *)bundle
suffix:(NSString *)suffix
extension:(NSString *)extension {
NSString *appBundlePath = [self.applicationBundle bundlePath];
bundle = [appBundlePath stringByAppendingPathComponent:bundle];
file = [file stringByAppendingFormat:@"%@%@", suffix, extension];
return [bundle stringByAppendingPathComponent:file];
}
- (NSString *)locateSettingsFile: (NSString *)file {
static NSString* const kIASKBundleFolder = @"Settings.bundle";
static NSString* const kIASKBundleFolderAlt = @"InAppSettings.bundle";
static NSString* const kIASKBundleLocaleFolderExtension = @".lproj";
// The file is searched in the following order:
//
// InAppSettings.bundle/FILE~DEVICE.inApp.plist
// InAppSettings.bundle/FILE.inApp.plist
// InAppSettings.bundle/FILE~DEVICE.plist
// InAppSettings.bundle/FILE.plist
// Settings.bundle/FILE~DEVICE.inApp.plist
// Settings.bundle/FILE.inApp.plist
// Settings.bundle/FILE~DEVICE.plist
// Settings.bundle/FILE.plist
//
// where DEVICE is either "iphone" or "ipad" depending on the current
// interface idiom.
//
// Settings.app uses the ~DEVICE suffixes since iOS 4.0. There are some
// differences from this implementation:
// - For an iPhone-only app running on iPad, Settings.app will not use the
// ~iphone suffix. There is no point in using these suffixes outside
// of universal apps anyway.
// - This implementation uses the device suffixes on iOS 3.x as well.
// - also check current locale (short only)
NSArray *settingsBundleNames = @[kIASKBundleFolderAlt, kIASKBundleFolder];
NSArray *extensions = @[@".inApp.plist", @".plist"];
NSArray *plattformSuffixes = @[[self platformSuffixForInterfaceIdiom:UI_USER_INTERFACE_IDIOM()],
@""];
NSArray *languageFolders = @[[[[NSLocale preferredLanguages] objectAtIndex:0] stringByAppendingString:kIASKBundleLocaleFolderExtension],
@""];
NSString *path = nil;
NSFileManager *fileManager = [NSFileManager defaultManager];
for (NSString *settingsBundleName in settingsBundleNames) {
for (NSString *extension in extensions) {
for (NSString *platformSuffix in plattformSuffixes) {
for (NSString *languageFolder in languageFolders) {
path = [self file:file
withBundle:[settingsBundleName stringByAppendingPathComponent:languageFolder]
suffix:platformSuffix
extension:extension];
if ([fileManager fileExistsAtPath:path]) {
goto exitFromNestedLoop;
}
}
}
}
}
exitFromNestedLoop:
return path;
}
@end