/** * Copyright (c) 2015-present, Facebook, Inc. * All rights reserved. * * This source code is licensed under the BSD-style license found in the * LICENSE file in the root directory of this source tree. An additional grant * of patent rights can be found in the PATENTS file in the same directory. */ #import "RCTConvert.h" #import #import "RCTCache.h" #import "RCTDefines.h" @implementation RCTConvert void RCTLogConvertError(id json, const char *type) { RCTLogError(@"JSON value '%@' of type '%@' cannot be converted to %s", json, [json classForCoder], type); } RCT_CONVERTER(id, id, self) RCT_CONVERTER(BOOL, BOOL, boolValue) RCT_NUMBER_CONVERTER(double, doubleValue) RCT_NUMBER_CONVERTER(float, floatValue) RCT_NUMBER_CONVERTER(int, intValue) RCT_NUMBER_CONVERTER(int64_t, longLongValue); RCT_NUMBER_CONVERTER(uint64_t, unsignedLongLongValue); RCT_NUMBER_CONVERTER(NSInteger, integerValue) RCT_NUMBER_CONVERTER(NSUInteger, unsignedIntegerValue) RCT_CUSTOM_CONVERTER(NSArray *, NSArray, [NSArray arrayWithArray:json]) RCT_CUSTOM_CONVERTER(NSSet *, NSSet, [NSSet setWithArray:json]) RCT_CUSTOM_CONVERTER(NSDictionary *, NSDictionary, [NSDictionary dictionaryWithDictionary:json]) RCT_CONVERTER(NSString *, NSString, description) + (NSNumber *)NSNumber:(id)json { if ([json isKindOfClass:[NSNumber class]]) { return json; } else if ([json isKindOfClass:[NSString class]]) { static NSNumberFormatter *formatter; static dispatch_once_t onceToken; dispatch_once(&onceToken, ^{ formatter = [[NSNumberFormatter alloc] init]; formatter.locale = [NSLocale localeWithLocaleIdentifier:@"en_US_POSIX"]; }); NSNumber *number = [formatter numberFromString:json]; if (!number) { RCTLogConvertError(json, "a number"); } return number; } else if (json && json != (id)kCFNull) { RCTLogConvertError(json, "a number"); } return nil; } + (NSData *)NSData:(id)json { // TODO: should we automatically decode base64 data? Probably not... return [[self NSString:json] dataUsingEncoding:NSUTF8StringEncoding]; } + (NSIndexSet *)NSIndexSet:(id)json { json = [self NSNumberArray:json]; NSMutableIndexSet *indexSet = [[NSMutableIndexSet alloc] init]; for (NSNumber *number in json) { NSInteger index = number.integerValue; if (RCT_DEBUG && index < 0) { RCTLogError(@"Invalid index value %zd. Indices must be positive.", index); } [indexSet addIndex:index]; } return indexSet; } + (NSURL *)NSURL:(id)json { NSString *path = [self NSString:json]; if (!path.length) { return nil; } @try { // NSURL has a history of crashing with bad input, so let's be safe NSURL *URL = [NSURL URLWithString:path]; if (URL.scheme) { // Was a well-formed absolute URL return URL; } // Check if it has a scheme if ([path rangeOfString:@"[a-zA-Z][a-zA-Z._-]+:" options:NSRegularExpressionSearch].location == 0) { path = [path stringByAddingPercentEscapesUsingEncoding:NSUTF8StringEncoding]; URL = [NSURL URLWithString:path]; if (URL) { return URL; } } // Assume that it's a local path path = [path stringByRemovingPercentEncoding]; if ([path hasPrefix:@"~"]) { // Path is inside user directory path = [path stringByExpandingTildeInPath]; } else if (![path isAbsolutePath]) { // Assume it's a resource path path = [[[NSBundle mainBundle] resourcePath] stringByAppendingPathComponent:path]; } return [NSURL fileURLWithPath:path]; } @catch (__unused NSException *e) { RCTLogConvertError(json, "a valid URL"); return nil; } } + (NSURLRequest *)NSURLRequest:(id)json { NSURL *URL = [self NSURL:json]; return URL ? [NSURLRequest requestWithURL:URL] : nil; } + (RCTFileURL *)RCTFileURL:(id)json { NSURL *fileURL = [self NSURL:json]; if (![fileURL isFileURL]) { RCTLogError(@"URI must be a local file, '%@' isn't.", fileURL); return nil; } if (![[NSFileManager defaultManager] fileExistsAtPath:fileURL.path]) { RCTLogError(@"File '%@' could not be found.", fileURL); return nil; } return fileURL; } + (NSDate *)NSDate:(id)json { if ([json isKindOfClass:[NSNumber class]]) { return [NSDate dateWithTimeIntervalSince1970:[self NSTimeInterval:json]]; } else if ([json isKindOfClass:[NSString class]]) { static NSDateFormatter *formatter; static dispatch_once_t onceToken; dispatch_once(&onceToken, ^{ formatter = [[NSDateFormatter alloc] init]; formatter.dateFormat = @"yyyy-MM-dd'T'HH:mm:ss.SSSZZZZZ"; formatter.locale = [NSLocale localeWithLocaleIdentifier:@"en_US_POSIX"]; formatter.timeZone = [NSTimeZone timeZoneWithName:@"UTC"]; }); NSDate *date = [formatter dateFromString:json]; if (!date) { RCTLogError(@"JSON String '%@' could not be interpreted as a date. " "Expected format: YYYY-MM-DD'T'HH:mm:ss.sssZ", json); } return date; } else if (json && json != (id)kCFNull) { RCTLogConvertError(json, "a date"); } return nil; } // JS Standard for time is milliseconds RCT_CUSTOM_CONVERTER(NSTimeInterval, NSTimeInterval, [self double:json] / 1000.0) // JS standard for time zones is minutes. RCT_CUSTOM_CONVERTER(NSTimeZone *, NSTimeZone, [NSTimeZone timeZoneForSecondsFromGMT:[self double:json] * 60.0]) NSNumber *RCTConvertEnumValue(const char *typeName, NSDictionary *mapping, NSNumber *defaultValue, id json) { if (!json || json == (id)kCFNull) { return defaultValue; } if ([json isKindOfClass:[NSNumber class]]) { NSArray *allValues = [mapping allValues]; if ([[mapping allValues] containsObject:json] || [json isEqual:defaultValue]) { return json; } RCTLogError(@"Invalid %s '%@'. should be one of: %@", typeName, json, allValues); return defaultValue; } if (![json isKindOfClass:[NSString class]]) { RCTLogError(@"Expected NSNumber or NSString for %s, received %@: %@", typeName, [json classForCoder], json); } id value = mapping[json]; if (!value && [json description].length > 0) { RCTLogError(@"Invalid %s '%@'. should be one of: %@", typeName, json, [[mapping allKeys] sortedArrayUsingSelector: @selector(caseInsensitiveCompare:)]); } return value ?: defaultValue; } NSNumber *RCTConvertMultiEnumValue(const char *typeName, NSDictionary *mapping, NSNumber *defaultValue, id json) { if ([json isKindOfClass:[NSArray class]]) { if ([json count] == 0) { return defaultValue; } long long result = 0; for (id arrayElement in json) { NSNumber *value = RCTConvertEnumValue(typeName, mapping, defaultValue, arrayElement); result |= [value longLongValue]; } return @(result); } return RCTConvertEnumValue(typeName, mapping, defaultValue, json); } RCT_ENUM_CONVERTER(NSTextAlignment, (@{ @"auto": @(NSTextAlignmentNatural), @"left": @(NSTextAlignmentLeft), @"center": @(NSTextAlignmentCenter), @"right": @(NSTextAlignmentRight), @"justify": @(NSTextAlignmentJustified), }), NSTextAlignmentNatural, integerValue) RCT_ENUM_CONVERTER(NSUnderlineStyle, (@{ @"solid": @(NSUnderlineStyleSingle), @"double": @(NSUnderlineStyleDouble), @"dotted": @(NSUnderlinePatternDot | NSUnderlineStyleSingle), @"dashed": @(NSUnderlinePatternDash | NSUnderlineStyleSingle), }), NSUnderlineStyleSingle, integerValue) RCT_ENUM_CONVERTER(RCTTextDecorationLineType, (@{ @"none": @(RCTTextDecorationLineTypeNone), @"underline": @(RCTTextDecorationLineTypeUnderline), @"line-through": @(RCTTextDecorationLineTypeStrikethrough), @"underline line-through": @(RCTTextDecorationLineTypeUnderlineStrikethrough), }), RCTTextDecorationLineTypeNone, integerValue) RCT_ENUM_CONVERTER(NSWritingDirection, (@{ @"auto": @(NSWritingDirectionNatural), @"ltr": @(NSWritingDirectionLeftToRight), @"rtl": @(NSWritingDirectionRightToLeft), }), NSWritingDirectionNatural, integerValue) RCT_ENUM_CONVERTER(UITextAutocapitalizationType, (@{ @"none": @(UITextAutocapitalizationTypeNone), @"words": @(UITextAutocapitalizationTypeWords), @"sentences": @(UITextAutocapitalizationTypeSentences), @"characters": @(UITextAutocapitalizationTypeAllCharacters) }), UITextAutocapitalizationTypeSentences, integerValue) RCT_ENUM_CONVERTER(UITextFieldViewMode, (@{ @"never": @(UITextFieldViewModeNever), @"while-editing": @(UITextFieldViewModeWhileEditing), @"unless-editing": @(UITextFieldViewModeUnlessEditing), @"always": @(UITextFieldViewModeAlways), }), UITextFieldViewModeNever, integerValue) RCT_ENUM_CONVERTER(UIKeyboardType, (@{ @"default": @(UIKeyboardTypeDefault), @"ascii-capable": @(UIKeyboardTypeASCIICapable), @"numbers-and-punctuation": @(UIKeyboardTypeNumbersAndPunctuation), @"url": @(UIKeyboardTypeURL), @"number-pad": @(UIKeyboardTypeNumberPad), @"phone-pad": @(UIKeyboardTypePhonePad), @"name-phone-pad": @(UIKeyboardTypeNamePhonePad), @"email-address": @(UIKeyboardTypeEmailAddress), @"decimal-pad": @(UIKeyboardTypeDecimalPad), @"twitter": @(UIKeyboardTypeTwitter), @"web-search": @(UIKeyboardTypeWebSearch), // Added for Android compatibility @"numeric": @(UIKeyboardTypeDecimalPad), }), UIKeyboardTypeDefault, integerValue) RCT_ENUM_CONVERTER(UIReturnKeyType, (@{ @"default": @(UIReturnKeyDefault), @"go": @(UIReturnKeyGo), @"google": @(UIReturnKeyGoogle), @"join": @(UIReturnKeyJoin), @"next": @(UIReturnKeyNext), @"route": @(UIReturnKeyRoute), @"search": @(UIReturnKeySearch), @"send": @(UIReturnKeySend), @"yahoo": @(UIReturnKeyYahoo), @"done": @(UIReturnKeyDone), @"emergency-call": @(UIReturnKeyEmergencyCall), }), UIReturnKeyDefault, integerValue) RCT_ENUM_CONVERTER(UIViewContentMode, (@{ @"scale-to-fill": @(UIViewContentModeScaleToFill), @"scale-aspect-fit": @(UIViewContentModeScaleAspectFit), @"scale-aspect-fill": @(UIViewContentModeScaleAspectFill), @"redraw": @(UIViewContentModeRedraw), @"center": @(UIViewContentModeCenter), @"top": @(UIViewContentModeTop), @"bottom": @(UIViewContentModeBottom), @"left": @(UIViewContentModeLeft), @"right": @(UIViewContentModeRight), @"top-left": @(UIViewContentModeTopLeft), @"top-right": @(UIViewContentModeTopRight), @"bottom-left": @(UIViewContentModeBottomLeft), @"bottom-right": @(UIViewContentModeBottomRight), // Cross-platform values @"cover": @(UIViewContentModeScaleAspectFill), @"contain": @(UIViewContentModeScaleAspectFit), @"stretch": @(UIViewContentModeScaleToFill), }), UIViewContentModeScaleAspectFill, integerValue) RCT_ENUM_CONVERTER(UIBarStyle, (@{ @"default": @(UIBarStyleDefault), @"black": @(UIBarStyleBlack), }), UIBarStyleDefault, integerValue) // TODO: normalise the use of w/width so we can do away with the alias values (#6566645) static void RCTConvertCGStructValue(const char *type, NSArray *fields, NSDictionary *aliases, CGFloat *result, id json) { NSUInteger count = fields.count; if ([json isKindOfClass:[NSArray class]]) { if (RCT_DEBUG && [json count] != count) { RCTLogError(@"Expected array with count %zd, but count is %zd: %@", count, [json count], json); } else { for (NSUInteger i = 0; i < count; i++) { result[i] = [RCTConvert CGFloat:json[i]]; } } } else if ([json isKindOfClass:[NSDictionary class]]) { if (aliases.count) { json = [json mutableCopy]; for (NSString *alias in aliases) { NSString *key = aliases[alias]; NSNumber *number = json[alias]; if (number) { RCTLogWarn(@"Using deprecated '%@' property for '%s'. Use '%@' instead.", alias, type, key); ((NSMutableDictionary *)json)[key] = number; } } } for (NSUInteger i = 0; i < count; i++) { result[i] = [RCTConvert CGFloat:json[fields[i]]]; } } else if (RCT_DEBUG && json && json != (id)kCFNull) { RCTLogConvertError(json, type); } } /** * This macro is used for creating converter functions for structs that consist * of a number of CGFloat properties, such as CGPoint, CGRect, etc. */ #define RCT_CGSTRUCT_CONVERTER(type, values, aliases) \ + (type)type:(id)json \ { \ static NSArray *fields; \ static dispatch_once_t onceToken; \ dispatch_once(&onceToken, ^{ \ fields = values; \ }); \ type result; \ RCTConvertCGStructValue(#type, fields, aliases, (CGFloat *)&result, json); \ return result; \ } RCT_CUSTOM_CONVERTER(CGFloat, CGFloat, [self double:json]) RCT_CGSTRUCT_CONVERTER(CGPoint, (@[@"x", @"y"]), (@{@"l": @"x", @"t": @"y"})) RCT_CGSTRUCT_CONVERTER(CGSize, (@[@"width", @"height"]), (@{@"w": @"width", @"h": @"height"})) RCT_CGSTRUCT_CONVERTER(CGRect, (@[@"x", @"y", @"width", @"height"]), (@{@"l": @"x", @"t": @"y", @"w": @"width", @"h": @"height"})) RCT_CGSTRUCT_CONVERTER(UIEdgeInsets, (@[@"top", @"left", @"bottom", @"right"]), nil) RCT_ENUM_CONVERTER(CGLineJoin, (@{ @"miter": @(kCGLineJoinMiter), @"round": @(kCGLineJoinRound), @"bevel": @(kCGLineJoinBevel), }), kCGLineJoinMiter, intValue) RCT_ENUM_CONVERTER(CGLineCap, (@{ @"butt": @(kCGLineCapButt), @"round": @(kCGLineCapRound), @"square": @(kCGLineCapSquare), }), kCGLineCapButt, intValue) RCT_CGSTRUCT_CONVERTER(CATransform3D, (@[ @"m11", @"m12", @"m13", @"m14", @"m21", @"m22", @"m23", @"m24", @"m31", @"m32", @"m33", @"m34", @"m41", @"m42", @"m43", @"m44" ]), nil) RCT_CGSTRUCT_CONVERTER(CGAffineTransform, (@[ @"a", @"b", @"c", @"d", @"tx", @"ty" ]), nil) + (UIColor *)UIColor:(id)json { // Check color cache static RCTCache *colorCache = nil; static dispatch_once_t onceToken; dispatch_once(&onceToken, ^{ colorCache = [[RCTCache alloc] init]; colorCache.countLimit = 128; }); UIColor *color = colorCache[json]; if (color) { return color; } if ([json isKindOfClass:[NSString class]]) { // Check named colors static NSDictionary *namedColors = nil; static dispatch_once_t onceToken; dispatch_once(&onceToken, ^{ namedColors = @{ // CSS colors @"aliceblue": @"#f0f8ff", @"antiquewhite": @"#faebd7", @"aqua": @"#00ffff", @"aquamarine": @"#7fffd4", @"azure": @"#f0ffff", @"beige": @"#f5f5dc", @"bisque": @"#ffe4c4", @"black": @"#000000", @"blanchedalmond": @"#ffebcd", @"blue": @"#0000ff", @"blueviolet": @"#8a2be2", @"brown": @"#a52a2a", @"burlywood": @"#deb887", @"cadetblue": @"#5f9ea0", @"chartreuse": @"#7fff00", @"chocolate": @"#d2691e", @"coral": @"#ff7f50", @"cornflowerblue": @"#6495ed", @"cornsilk": @"#fff8dc", @"crimson": @"#dc143c", @"cyan": @"#00ffff", @"darkblue": @"#00008b", @"darkcyan": @"#008b8b", @"darkgoldenrod": @"#b8860b", @"darkgray": @"#a9a9a9", @"darkgrey": @"#a9a9a9", @"darkgreen": @"#006400", @"darkkhaki": @"#bdb76b", @"darkmagenta": @"#8b008b", @"darkolivegreen": @"#556b2f", @"darkorange": @"#ff8c00", @"darkorchid": @"#9932cc", @"darkred": @"#8b0000", @"darksalmon": @"#e9967a", @"darkseagreen": @"#8fbc8f", @"darkslateblue": @"#483d8b", @"darkslategray": @"#2f4f4f", @"darkslategrey": @"#2f4f4f", @"darkturquoise": @"#00ced1", @"darkviolet": @"#9400d3", @"deeppink": @"#ff1493", @"deepskyblue": @"#00bfff", @"dimgray": @"#696969", @"dimgrey": @"#696969", @"dodgerblue": @"#1e90ff", @"firebrick": @"#b22222", @"floralwhite": @"#fffaf0", @"forestgreen": @"#228b22", @"fuchsia": @"#ff00ff", @"gainsboro": @"#dcdcdc", @"ghostwhite": @"#f8f8ff", @"gold": @"#ffd700", @"goldenrod": @"#daa520", @"gray": @"#808080", @"grey": @"#808080", @"green": @"#008000", @"greenyellow": @"#adff2f", @"honeydew": @"#f0fff0", @"hotpink": @"#ff69b4", @"indianred": @"#cd5c5c", @"indigo": @"#4b0082", @"ivory": @"#fffff0", @"khaki": @"#f0e68c", @"lavender": @"#e6e6fa", @"lavenderblush": @"#fff0f5", @"lawngreen": @"#7cfc00", @"lemonchiffon": @"#fffacd", @"lightblue": @"#add8e6", @"lightcoral": @"#f08080", @"lightcyan": @"#e0ffff", @"lightgoldenrodyellow": @"#fafad2", @"lightgray": @"#d3d3d3", @"lightgrey": @"#d3d3d3", @"lightgreen": @"#90ee90", @"lightpink": @"#ffb6c1", @"lightsalmon": @"#ffa07a", @"lightseagreen": @"#20b2aa", @"lightskyblue": @"#87cefa", @"lightslategray": @"#778899", @"lightslategrey": @"#778899", @"lightsteelblue": @"#b0c4de", @"lightyellow": @"#ffffe0", @"lime": @"#00ff00", @"limegreen": @"#32cd32", @"linen": @"#faf0e6", @"magenta": @"#ff00ff", @"maroon": @"#800000", @"mediumaquamarine": @"#66cdaa", @"mediumblue": @"#0000cd", @"mediumorchid": @"#ba55d3", @"mediumpurple": @"#9370db", @"mediumseagreen": @"#3cb371", @"mediumslateblue": @"#7b68ee", @"mediumspringgreen": @"#00fa9a", @"mediumturquoise": @"#48d1cc", @"mediumvioletred": @"#c71585", @"midnightblue": @"#191970", @"mintcream": @"#f5fffa", @"mistyrose": @"#ffe4e1", @"moccasin": @"#ffe4b5", @"navajowhite": @"#ffdead", @"navy": @"#000080", @"oldlace": @"#fdf5e6", @"olive": @"#808000", @"olivedrab": @"#6b8e23", @"orange": @"#ffa500", @"orangered": @"#ff4500", @"orchid": @"#da70d6", @"palegoldenrod": @"#eee8aa", @"palegreen": @"#98fb98", @"paleturquoise": @"#afeeee", @"palevioletred": @"#db7093", @"papayawhip": @"#ffefd5", @"peachpuff": @"#ffdab9", @"peru": @"#cd853f", @"pink": @"#ffc0cb", @"plum": @"#dda0dd", @"powderblue": @"#b0e0e6", @"purple": @"#800080", @"rebeccapurple": @"#663399", @"red": @"#ff0000", @"rosybrown": @"#bc8f8f", @"royalblue": @"#4169e1", @"saddlebrown": @"#8b4513", @"salmon": @"#fa8072", @"sandybrown": @"#f4a460", @"seagreen": @"#2e8b57", @"seashell": @"#fff5ee", @"sienna": @"#a0522d", @"silver": @"#c0c0c0", @"skyblue": @"#87ceeb", @"slateblue": @"#6a5acd", @"slategray": @"#708090", @"slategrey": @"#708090", @"snow": @"#fffafa", @"springgreen": @"#00ff7f", @"steelblue": @"#4682b4", @"tan": @"#d2b48c", @"teal": @"#008080", @"thistle": @"#d8bfd8", @"tomato": @"#ff6347", @"turquoise": @"#40e0d0", @"violet": @"#ee82ee", @"wheat": @"#f5deb3", @"white": @"#ffffff", @"whitesmoke": @"#f5f5f5", @"yellow": @"#ffff00", @"yellowgreen": @"#9acd32", // Nonstandard color extensions @"transparent": @"rgba(0,0,0,0)", }; }); NSString *colorString = namedColors[json]; if (!colorString) { colorString = json; } // Parse color double red = 0, green = 0, blue = 0; double alpha = 1.0; if ([colorString hasPrefix:@"#"]) { uint32_t redInt = 0, greenInt = 0, blueInt = 0; if (colorString.length == 4) { // 3 digit hex sscanf([colorString UTF8String], "#%01x%01x%01x", &redInt, &greenInt, &blueInt); // expand to 6 digit hex red = redInt | (redInt << 4); green = greenInt | (greenInt << 4); blue = blueInt | (blueInt << 4); } else if (colorString.length == 7) { // 6 digit hex sscanf(colorString.UTF8String, "#%02x%02x%02x", &redInt, &greenInt, &blueInt); red = redInt; green = greenInt; blue = blueInt; } else { RCTLogError(@"Invalid hex color %@. Hex colors should be 3 or 6 digits long.", colorString); alpha = -1; } } else if ([colorString hasPrefix:@"rgba("]) { sscanf(colorString.UTF8String, "rgba(%lf,%lf,%lf,%lf)", &red, &green, &blue, &alpha); } else if ([colorString hasPrefix:@"rgb("]) { sscanf(colorString.UTF8String, "rgb(%lf,%lf,%lf)", &red, &green, &blue); } else { RCTLogError(@"Unrecognized color format '%@', must be one of #hex|rgba|rgb or a valid CSS color name.", colorString); alpha = -1; } if (alpha < 0) { RCTLogError(@"Invalid color string '%@'", colorString); } else { color = [UIColor colorWithRed:red / 255.0 green:green / 255.0 blue:blue / 255.0 alpha:alpha]; } } else if ([json isKindOfClass:[NSArray class]]) { if ([json count] < 3 || [json count] > 4) { RCTLogError(@"Expected array with count 3 or 4, but count is %zd: %@", [json count], json); } else { // Color array color = [UIColor colorWithRed:[self CGFloat:json[0]] green:[self CGFloat:json[1]] blue:[self CGFloat:json[2]] alpha:[json count] > 3 ? [self CGFloat:json[3]] : 1]; } } else if ([json isKindOfClass:[NSDictionary class]]) { // Color dictionary color = [UIColor colorWithRed:[self CGFloat:json[@"r"]] green:[self CGFloat:json[@"g"]] blue:[self CGFloat:json[@"b"]] alpha:[self CGFloat:json[@"a"] ?: @1]]; } else if (RCT_DEBUG && json && json != (id)kCFNull) { RCTLogConvertError(json, "a color"); } // Default color if (!color) { color = [UIColor whiteColor]; } // Cache and return if (json) { colorCache[json] = color; } return color; } + (CGColorRef)CGColor:(id)json { return [self UIColor:json].CGColor; } + (UIImage *)UIImage:(id)json { // TODO: we might as well cache the result of these checks (and possibly the // image itself) so as to reduce overhead on subsequent checks of the same input if (!json || json == (id)kCFNull) { return nil; } UIImage *image; NSString *path; CGFloat scale = 0.0; if ([json isKindOfClass:[NSString class]]) { path = json; } else if ([json isKindOfClass:[NSDictionary class]]) { path = [self NSString:json[@"uri"]]; scale = [self CGFloat:json[@"scale"]]; } else { RCTLogConvertError(json, "an image"); } NSURL *URL = [self NSURL:path]; NSString *scheme = [URL.scheme lowercaseString]; if ([scheme isEqualToString:@"file"]) { if ([NSThread currentThread] == [NSThread mainThread]) { // Image may reside inside a .car file, in which case we have no choice // but to use +[UIImage imageNamed] - but this method isn't thread safe image = [UIImage imageNamed:path]; } if (!image) { // Attempt to load from the file system if ([path pathExtension].length == 0) { path = [path stringByAppendingPathExtension:@"png"]; } image = [UIImage imageWithContentsOfFile:path]; } // We won't warn about nil images because there are legitimate cases // where we find out if a string is an image by using this method, but // we do enforce thread-safe API usage with the following check if (RCT_DEBUG && !image && [UIImage imageNamed:path]) { RCTAssertMainThread(); } } else if ([scheme isEqualToString:@"data"]) { image = [UIImage imageWithData:[NSData dataWithContentsOfURL:URL]]; } else { RCTLogConvertError(json, "an image. Only local files or data URIs are supported"); } if (scale > 0) { image = [UIImage imageWithCGImage:image.CGImage scale:scale orientation:image.imageOrientation]; } return image; } + (CGImageRef)CGImage:(id)json { return [self UIImage:json].CGImage; } #if !defined(__IPHONE_8_2) || __IPHONE_OS_VERSION_MIN_REQUIRED < __IPHONE_8_2 // These constants are defined in iPhone SDK 8.2, but the app cannot run on // iOS < 8.2 unless we redefine them here. If you target iOS 8.2 or above // as a base target, the standard constants will be used instead. #define UIFontWeightUltraLight -0.8 #define UIFontWeightThin -0.6 #define UIFontWeightLight -0.4 #define UIFontWeightRegular 0 #define UIFontWeightMedium 0.23 #define UIFontWeightSemibold 0.3 #define UIFontWeightBold 0.4 #define UIFontWeightHeavy 0.56 #define UIFontWeightBlack 0.62 #endif typedef CGFloat RCTFontWeight; RCT_ENUM_CONVERTER(RCTFontWeight, (@{ @"normal": @(UIFontWeightRegular), @"bold": @(UIFontWeightBold), @"100": @(UIFontWeightUltraLight), @"200": @(UIFontWeightThin), @"300": @(UIFontWeightLight), @"400": @(UIFontWeightRegular), @"500": @(UIFontWeightMedium), @"600": @(UIFontWeightSemibold), @"700": @(UIFontWeightBold), @"800": @(UIFontWeightHeavy), @"900": @(UIFontWeightBlack), }), UIFontWeightRegular, doubleValue) typedef BOOL RCTFontStyle; RCT_ENUM_CONVERTER(RCTFontStyle, (@{ @"normal": @NO, @"italic": @YES, @"oblique": @YES, }), NO, boolValue) static RCTFontWeight RCTWeightOfFont(UIFont *font) { NSDictionary *traits = [font.fontDescriptor objectForKey:UIFontDescriptorTraitsAttribute]; return [traits[UIFontWeightTrait] doubleValue]; } static BOOL RCTFontIsItalic(UIFont *font) { NSDictionary *traits = [font.fontDescriptor objectForKey:UIFontDescriptorTraitsAttribute]; UIFontDescriptorSymbolicTraits symbolicTraits = [traits[UIFontSymbolicTrait] unsignedIntValue]; return (symbolicTraits & UIFontDescriptorTraitItalic) != 0; } static BOOL RCTFontIsCondensed(UIFont *font) { NSDictionary *traits = [font.fontDescriptor objectForKey:UIFontDescriptorTraitsAttribute]; UIFontDescriptorSymbolicTraits symbolicTraits = [traits[UIFontSymbolicTrait] unsignedIntValue]; return (symbolicTraits & UIFontDescriptorTraitCondensed) != 0; } + (UIFont *)UIFont:(id)json { json = [self NSDictionary:json]; return [self UIFont:nil withFamily:json[@"fontFamily"] size:json[@"fontSize"] weight:json[@"fontWeight"] style:json[@"fontStyle"]]; } + (UIFont *)UIFont:(UIFont *)font withSize:(id)json { return [self UIFont:font withFamily:nil size:json weight:nil style:nil]; } + (UIFont *)UIFont:(UIFont *)font withWeight:(id)json { return [self UIFont:font withFamily:nil size:nil weight:json style:nil]; } + (UIFont *)UIFont:(UIFont *)font withStyle:(id)json { return [self UIFont:font withFamily:nil size:nil weight:nil style:json]; } + (UIFont *)UIFont:(UIFont *)font withFamily:(id)json { return [self UIFont:font withFamily:json size:nil weight:nil style:nil]; } + (UIFont *)UIFont:(UIFont *)font withFamily:(id)family size:(id)size weight:(id)weight style:(id)style { // Defaults NSString *const RCTDefaultFontFamily = @"System"; NSString *const RCTIOS8SystemFontFamily = @"Helvetica Neue"; const RCTFontWeight RCTDefaultFontWeight = UIFontWeightRegular; const CGFloat RCTDefaultFontSize = 14; // Initialize properties to defaults CGFloat fontSize = RCTDefaultFontSize; RCTFontWeight fontWeight = RCTDefaultFontWeight; NSString *familyName = RCTDefaultFontFamily; BOOL isItalic = NO; BOOL isCondensed = NO; if (font) { familyName = font.familyName ?: RCTDefaultFontFamily; fontSize = font.pointSize ?: RCTDefaultFontSize; fontWeight = RCTWeightOfFont(font); isItalic = RCTFontIsItalic(font); isCondensed = RCTFontIsCondensed(font); } // Get font attributes fontSize = [self CGFloat:size] ?: fontSize; familyName = [self NSString:family] ?: familyName; isItalic = style ? [self RCTFontStyle:style] : isItalic; fontWeight = weight ? [self RCTFontWeight:weight] : fontWeight; // Handle system font as special case. This ensures that we preserve // the specific metrics of the standard system font as closely as possible. if ([familyName isEqual:RCTDefaultFontFamily]) { if ([UIFont respondsToSelector:@selector(systemFontOfSize:weight:)]) { font = [UIFont systemFontOfSize:fontSize weight:fontWeight]; if (isItalic || isCondensed) { UIFontDescriptor *fontDescriptor = [font fontDescriptor]; UIFontDescriptorSymbolicTraits symbolicTraits = fontDescriptor.symbolicTraits; if (isItalic) { symbolicTraits |= UIFontDescriptorTraitItalic; } if (isCondensed) { symbolicTraits |= UIFontDescriptorTraitCondensed; } fontDescriptor = [fontDescriptor fontDescriptorWithSymbolicTraits:symbolicTraits]; font = [UIFont fontWithDescriptor:fontDescriptor size:fontSize]; } return font; } else { // systemFontOfSize:weight: isn't available prior to iOS 8.2, so we // fall back to finding the correct font manually, by linear search. familyName = RCTIOS8SystemFontFamily; } } // Gracefully handle being given a font name rather than font family, for // example: "Helvetica Light Oblique" rather than just "Helvetica". if ([UIFont fontNamesForFamilyName:familyName].count == 0) { font = [UIFont fontWithName:familyName size:fontSize]; if (font) { // It's actually a font name, not a font family name, // but we'll do what was meant, not what was said. familyName = font.familyName; fontWeight = weight ? fontWeight : RCTWeightOfFont(font); isItalic = style ? isItalic : RCTFontIsItalic(font); isCondensed = RCTFontIsCondensed(font); } else { // Not a valid font or family RCTLogError(@"Unrecognized font family '%@'", familyName); if ([UIFont respondsToSelector:@selector(systemFontOfSize:weight:)]) { font = [UIFont systemFontOfSize:fontSize weight:fontWeight]; } else if (fontWeight > UIFontWeightRegular) { font = [UIFont boldSystemFontOfSize:fontSize]; } else { font = [UIFont systemFontOfSize:fontSize]; } } } // Get the closest font that matches the given weight for the fontFamily UIFont *bestMatch = font; CGFloat closestWeight = INFINITY; for (NSString *name in [UIFont fontNamesForFamilyName:familyName]) { UIFont *match = [UIFont fontWithName:name size:fontSize]; if (isItalic == RCTFontIsItalic(match) && isCondensed == RCTFontIsCondensed(match)) { CGFloat testWeight = RCTWeightOfFont(match); if (ABS(testWeight - fontWeight) < ABS(closestWeight - fontWeight)) { bestMatch = match; closestWeight = testWeight; } } } return bestMatch; } NSArray *RCTConvertArrayValue(SEL type, id json) { __block BOOL copy = NO; __block NSArray *values = json = [RCTConvert NSArray:json]; [json enumerateObjectsUsingBlock:^(id jsonValue, NSUInteger idx, __unused BOOL *stop) { id value = ((id(*)(Class, SEL, id))objc_msgSend)([RCTConvert class], type, jsonValue); if (copy) { if (value) { [(NSMutableArray *)values addObject:value]; } } else if (value != jsonValue) { // Converted value is different, so we'll need to copy the array values = [[NSMutableArray alloc] initWithCapacity:values.count]; for (NSUInteger i = 0; i < idx; i++) { [(NSMutableArray *)values addObject:json[i]]; } if (value) { [(NSMutableArray *)values addObject:value]; } copy = YES; } }]; return values; } RCT_ARRAY_CONVERTER(NSString) RCT_ARRAY_CONVERTER(NSDictionary) RCT_ARRAY_CONVERTER(NSURL) RCT_ARRAY_CONVERTER(RCTFileURL) RCT_ARRAY_CONVERTER(NSNumber) RCT_ARRAY_CONVERTER(UIColor) // Can't use RCT_ARRAY_CONVERTER due to bridged cast + (NSArray *)CGColorArray:(id)json { NSMutableArray *colors = [[NSMutableArray alloc] init]; for (id value in [self NSArray:json]) { [colors addObject:(__bridge id)[self CGColor:value]]; } return colors; } static id RCTConvertPropertyListValue(id json) { if (!json || json == (id)kCFNull) { return nil; } if ([json isKindOfClass:[NSDictionary class]]) { __block BOOL copy = NO; NSMutableDictionary *values = [[NSMutableDictionary alloc] initWithCapacity:[json count]]; [json enumerateKeysAndObjectsUsingBlock:^(NSString *key, id jsonValue, __unused BOOL *stop) { id value = RCTConvertPropertyListValue(jsonValue); if (value) { values[key] = value; } copy |= value != jsonValue; }]; return copy ? values : json; } if ([json isKindOfClass:[NSArray class]]) { __block BOOL copy = NO; __block NSArray *values = json; [json enumerateObjectsUsingBlock:^(id jsonValue, NSUInteger idx, __unused BOOL *stop) { id value = RCTConvertPropertyListValue(jsonValue); if (copy) { if (value) { [(NSMutableArray *)values addObject:value]; } } else if (value != jsonValue) { // Converted value is different, so we'll need to copy the array values = [[NSMutableArray alloc] initWithCapacity:values.count]; for (NSUInteger i = 0; i < idx; i++) { [(NSMutableArray *)values addObject:json[i]]; } if (value) { [(NSMutableArray *)values addObject:value]; } copy = YES; } }]; return values; } // All other JSON types are supported by property lists return json; } + (NSPropertyList)NSPropertyList:(id)json { return RCTConvertPropertyListValue(json); } RCT_ENUM_CONVERTER(css_backface_visibility_t, (@{ @"hidden": @NO, @"visible": @YES }), YES, boolValue) RCT_ENUM_CONVERTER(css_clip_t, (@{ @"hidden": @YES, @"visible": @NO }), NO, boolValue) RCT_ENUM_CONVERTER(css_flex_direction_t, (@{ @"row": @(CSS_FLEX_DIRECTION_ROW), @"column": @(CSS_FLEX_DIRECTION_COLUMN) }), CSS_FLEX_DIRECTION_COLUMN, intValue) RCT_ENUM_CONVERTER(css_justify_t, (@{ @"flex-start": @(CSS_JUSTIFY_FLEX_START), @"flex-end": @(CSS_JUSTIFY_FLEX_END), @"center": @(CSS_JUSTIFY_CENTER), @"space-between": @(CSS_JUSTIFY_SPACE_BETWEEN), @"space-around": @(CSS_JUSTIFY_SPACE_AROUND) }), CSS_JUSTIFY_FLEX_START, intValue) RCT_ENUM_CONVERTER(css_align_t, (@{ @"flex-start": @(CSS_ALIGN_FLEX_START), @"flex-end": @(CSS_ALIGN_FLEX_END), @"center": @(CSS_ALIGN_CENTER), @"auto": @(CSS_ALIGN_AUTO), @"stretch": @(CSS_ALIGN_STRETCH) }), CSS_ALIGN_FLEX_START, intValue) RCT_ENUM_CONVERTER(css_position_type_t, (@{ @"absolute": @(CSS_POSITION_ABSOLUTE), @"relative": @(CSS_POSITION_RELATIVE) }), CSS_POSITION_RELATIVE, intValue) RCT_ENUM_CONVERTER(css_wrap_type_t, (@{ @"wrap": @(CSS_WRAP), @"nowrap": @(CSS_NOWRAP) }), CSS_NOWRAP, intValue) RCT_ENUM_CONVERTER(RCTPointerEvents, (@{ @"none": @(RCTPointerEventsNone), @"box-only": @(RCTPointerEventsBoxOnly), @"box-none": @(RCTPointerEventsBoxNone), @"auto": @(RCTPointerEventsUnspecified) }), RCTPointerEventsUnspecified, integerValue) RCT_ENUM_CONVERTER(RCTAnimationType, (@{ @"spring": @(RCTAnimationTypeSpring), @"linear": @(RCTAnimationTypeLinear), @"easeIn": @(RCTAnimationTypeEaseIn), @"easeOut": @(RCTAnimationTypeEaseOut), @"easeInEaseOut": @(RCTAnimationTypeEaseInEaseOut), @"keyboard": @(RCTAnimationTypeKeyboard), }), RCTAnimationTypeEaseInEaseOut, integerValue) @end BOOL RCTSetProperty(id target, NSString *keyPath, SEL type, id json) { // Split keypath NSArray *parts = [keyPath componentsSeparatedByString:@"."]; NSString *key = [parts lastObject]; for (NSUInteger i = 0; i < parts.count - 1; i++) { target = [target valueForKey:parts[i]]; if (!target) { return NO; } } // Get property setter SEL setter = NSSelectorFromString([NSString stringWithFormat:@"set%@%@:", [[key substringToIndex:1] uppercaseString], [key substringFromIndex:1]]); // Fail early if (![target respondsToSelector:setter]) { return NO; } @try { NSMethodSignature *signature = [RCTConvert methodSignatureForSelector:type]; switch (signature.methodReturnType[0]) { #define RCT_SET_CASE(_value, _type) \ case _value: { \ _type (*convert)(id, SEL, id) = (typeof(convert))objc_msgSend; \ void (*set)(id, SEL, _type) = (typeof(set))objc_msgSend; \ set(target, setter, convert([RCTConvert class], type, json)); \ break; \ } RCT_SET_CASE(':', SEL) RCT_SET_CASE('*', const char *) RCT_SET_CASE('c', char) RCT_SET_CASE('C', unsigned char) RCT_SET_CASE('s', short) RCT_SET_CASE('S', unsigned short) RCT_SET_CASE('i', int) RCT_SET_CASE('I', unsigned int) RCT_SET_CASE('l', long) RCT_SET_CASE('L', unsigned long) RCT_SET_CASE('q', long long) RCT_SET_CASE('Q', unsigned long long) RCT_SET_CASE('f', float) RCT_SET_CASE('d', double) RCT_SET_CASE('B', BOOL) RCT_SET_CASE('^', void *) case '@': { id (*convert)(id, SEL, id) = (typeof(convert))objc_msgSend; void (*set)(id, SEL, id) = (typeof(set))objc_msgSend; set(target, setter, convert([RCTConvert class], type, json)); break; } case '{': default: { // Get converted value void *value = malloc(signature.methodReturnLength); NSInvocation *invocation = [NSInvocation invocationWithMethodSignature:signature]; [invocation setTarget:[RCTConvert class]]; [invocation setSelector:type]; [invocation setArgument:&json atIndex:2]; [invocation invoke]; [invocation getReturnValue:value]; // Set converted value signature = [target methodSignatureForSelector:setter]; invocation = [NSInvocation invocationWithMethodSignature:signature]; [invocation setArgument:&setter atIndex:1]; [invocation setArgument:value atIndex:2]; [invocation invokeWithTarget:target]; free(value); break; } } return YES; } @catch (NSException *exception) { RCTLogError(@"Exception thrown while attempting to set property '%@' of \ '%@' with value '%@': %@", key, [target class], json, exception); return NO; } } BOOL RCTCopyProperty(id target, id source, NSString *keyPath) { // Split keypath NSArray *parts = [keyPath componentsSeparatedByString:@"."]; NSString *key = [parts lastObject]; for (NSUInteger i = 0; i < parts.count - 1; i++) { source = [source valueForKey:parts[i]]; target = [target valueForKey:parts[i]]; if (!source || !target) { return NO; } } // Get property getter SEL getter = NSSelectorFromString(key); // Get property setter SEL setter = NSSelectorFromString([NSString stringWithFormat:@"set%@%@:", [[key substringToIndex:1] uppercaseString], [key substringFromIndex:1]]); // Fail early if (![source respondsToSelector:getter] || ![target respondsToSelector:setter]) { return NO; } NSMethodSignature *signature = [source methodSignatureForSelector:getter]; switch (signature.methodReturnType[0]) { #define RCT_COPY_CASE(_value, _type) \ case _value: { \ _type (*get)(id, SEL) = (typeof(get))objc_msgSend; \ void (*set)(id, SEL, _type) = (typeof(set))objc_msgSend; \ set(target, setter, get(source, getter)); \ break; \ } RCT_COPY_CASE(':', SEL) RCT_COPY_CASE('*', const char *) RCT_COPY_CASE('c', char) RCT_COPY_CASE('C', unsigned char) RCT_COPY_CASE('s', short) RCT_COPY_CASE('S', unsigned short) RCT_COPY_CASE('i', int) RCT_COPY_CASE('I', unsigned int) RCT_COPY_CASE('l', long) RCT_COPY_CASE('L', unsigned long) RCT_COPY_CASE('q', long long) RCT_COPY_CASE('Q', unsigned long long) RCT_COPY_CASE('f', float) RCT_COPY_CASE('d', double) RCT_COPY_CASE('B', BOOL) RCT_COPY_CASE('^', void *) case '@': { id (*get)(id, SEL) = (typeof(get))objc_msgSend; void (*set)(id, SEL, id) = (typeof(set))objc_msgSend; set(target, setter, get(source, getter)); break; } case '{': default: { // Get value void *value = malloc(signature.methodReturnLength); NSInvocation *invocation = [NSInvocation invocationWithMethodSignature:signature]; [invocation setArgument:&getter atIndex:1]; [invocation invokeWithTarget:source]; [invocation getReturnValue:value]; // Set value signature = [target methodSignatureForSelector:setter]; invocation = [NSInvocation invocationWithMethodSignature:signature]; [invocation setArgument:&setter atIndex:1]; [invocation setArgument:value atIndex:2]; [invocation invokeWithTarget:target]; free(value); break; } } return YES; }