2015-03-23 15:07:33 -07:00
|
|
|
/**
|
|
|
|
* 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.
|
|
|
|
*/
|
2015-01-29 17:10:49 -08:00
|
|
|
|
|
|
|
#import "RCTUtils.h"
|
|
|
|
|
|
|
|
#import <mach/mach_time.h>
|
|
|
|
#import <objc/runtime.h>
|
2015-03-10 19:11:28 -07:00
|
|
|
|
2015-01-29 17:10:49 -08:00
|
|
|
#import <UIKit/UIKit.h>
|
|
|
|
|
2015-03-10 19:11:28 -07:00
|
|
|
#import <CommonCrypto/CommonCrypto.h>
|
|
|
|
|
|
|
|
#import "RCTLog.h"
|
|
|
|
|
2015-01-29 17:10:49 -08:00
|
|
|
NSString *RCTJSONStringify(id jsonObject, NSError **error)
|
|
|
|
{
|
|
|
|
NSData *jsonData = [NSJSONSerialization dataWithJSONObject:jsonObject options:0 error:error];
|
|
|
|
return jsonData ? [[NSString alloc] initWithData:jsonData encoding:NSUTF8StringEncoding] : nil;
|
|
|
|
}
|
|
|
|
|
|
|
|
id RCTJSONParse(NSString *jsonString, NSError **error)
|
|
|
|
{
|
2015-03-10 19:11:28 -07:00
|
|
|
if (!jsonString) {
|
|
|
|
return nil;
|
|
|
|
}
|
|
|
|
NSData *jsonData = [jsonString dataUsingEncoding:NSUTF8StringEncoding allowLossyConversion:NO];
|
|
|
|
if (!jsonData) {
|
|
|
|
jsonData = [jsonString dataUsingEncoding:NSUTF8StringEncoding allowLossyConversion:YES];
|
2015-03-24 19:34:12 -07:00
|
|
|
if (jsonData) {
|
|
|
|
RCTLogWarn(@"RCTJSONParse received the following string, which could not be losslessly converted to UTF8 data: '%@'", jsonString);
|
|
|
|
} else {
|
|
|
|
// If our backup conversion fails, log the issue so we can see what strings are causing this (t6452813)
|
|
|
|
RCTLogError(@"RCTJSONParse received the following string, which could not be converted to UTF8 data: '%@'", jsonString);
|
|
|
|
return nil;
|
|
|
|
}
|
2015-03-10 19:11:28 -07:00
|
|
|
}
|
2015-01-29 17:10:49 -08:00
|
|
|
return [NSJSONSerialization JSONObjectWithData:jsonData options:NSJSONReadingAllowFragments error:error];
|
|
|
|
}
|
|
|
|
|
|
|
|
NSString *RCTMD5Hash(NSString *string)
|
|
|
|
{
|
|
|
|
const char *str = [string UTF8String];
|
|
|
|
unsigned char result[CC_MD5_DIGEST_LENGTH];
|
|
|
|
CC_MD5(str, (CC_LONG)strlen(str), result);
|
|
|
|
|
|
|
|
return [NSString stringWithFormat:@"%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x",
|
|
|
|
result[0], result[1], result[2], result[3],
|
|
|
|
result[4], result[5], result[6], result[7],
|
|
|
|
result[8], result[9], result[10], result[11],
|
|
|
|
result[12], result[13], result[14], result[15]
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
|
|
|
CGFloat RCTScreenScale()
|
|
|
|
{
|
2015-02-03 16:15:20 -08:00
|
|
|
static CGFloat scale;
|
2015-01-29 17:10:49 -08:00
|
|
|
static dispatch_once_t onceToken;
|
|
|
|
dispatch_once(&onceToken, ^{
|
|
|
|
if (![NSThread isMainThread]) {
|
|
|
|
dispatch_sync(dispatch_get_main_queue(), ^{
|
|
|
|
scale = [UIScreen mainScreen].scale;
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
scale = [UIScreen mainScreen].scale;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
return scale;
|
|
|
|
}
|
|
|
|
|
2015-02-03 16:15:20 -08:00
|
|
|
CGSize RCTScreenSize()
|
|
|
|
{
|
|
|
|
static CGSize size;
|
|
|
|
static dispatch_once_t onceToken;
|
|
|
|
dispatch_once(&onceToken, ^{
|
|
|
|
if (![NSThread isMainThread]) {
|
|
|
|
dispatch_sync(dispatch_get_main_queue(), ^{
|
|
|
|
size = [UIScreen mainScreen].bounds.size;
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
size = [UIScreen mainScreen].bounds.size;
|
|
|
|
}
|
|
|
|
});
|
2015-03-10 19:11:28 -07:00
|
|
|
|
2015-02-03 16:15:20 -08:00
|
|
|
return size;
|
|
|
|
}
|
|
|
|
|
2015-01-29 17:10:49 -08:00
|
|
|
CGFloat RCTRoundPixelValue(CGFloat value)
|
|
|
|
{
|
|
|
|
CGFloat scale = RCTScreenScale();
|
|
|
|
return round(value * scale) / scale;
|
|
|
|
}
|
|
|
|
|
|
|
|
CGFloat RCTCeilPixelValue(CGFloat value)
|
|
|
|
{
|
|
|
|
CGFloat scale = RCTScreenScale();
|
|
|
|
return ceil(value * scale) / scale;
|
|
|
|
}
|
|
|
|
|
|
|
|
CGFloat RCTFloorPixelValue(CGFloat value)
|
|
|
|
{
|
|
|
|
CGFloat scale = RCTScreenScale();
|
|
|
|
return floor(value * scale) / scale;
|
|
|
|
}
|
|
|
|
|
|
|
|
NSTimeInterval RCTTGetAbsoluteTime(void)
|
|
|
|
{
|
|
|
|
static struct mach_timebase_info tb_info = {0};
|
|
|
|
static dispatch_once_t onceToken;
|
|
|
|
dispatch_once(&onceToken, ^{
|
|
|
|
int ret = mach_timebase_info(&tb_info);
|
|
|
|
assert(0 == ret);
|
|
|
|
});
|
2015-03-10 19:11:28 -07:00
|
|
|
|
2015-01-29 17:10:49 -08:00
|
|
|
uint64_t timeInNanoseconds = (mach_absolute_time() * tb_info.numer) / tb_info.denom;
|
|
|
|
return ((NSTimeInterval)timeInNanoseconds) / 1000000;
|
|
|
|
}
|
|
|
|
|
|
|
|
void RCTSwapClassMethods(Class cls, SEL original, SEL replacement)
|
|
|
|
{
|
|
|
|
Method originalMethod = class_getClassMethod(cls, original);
|
|
|
|
IMP originalImplementation = method_getImplementation(originalMethod);
|
|
|
|
const char *originalArgTypes = method_getTypeEncoding(originalMethod);
|
2015-03-10 19:11:28 -07:00
|
|
|
|
2015-01-29 17:10:49 -08:00
|
|
|
Method replacementMethod = class_getClassMethod(cls, replacement);
|
|
|
|
IMP replacementImplementation = method_getImplementation(replacementMethod);
|
|
|
|
const char *replacementArgTypes = method_getTypeEncoding(replacementMethod);
|
2015-03-10 19:11:28 -07:00
|
|
|
|
2015-01-29 17:10:49 -08:00
|
|
|
if (class_addMethod(cls, original, replacementImplementation, replacementArgTypes))
|
|
|
|
{
|
|
|
|
class_replaceMethod(cls, replacement, originalImplementation, originalArgTypes);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
method_exchangeImplementations(originalMethod, replacementMethod);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void RCTSwapInstanceMethods(Class cls, SEL original, SEL replacement)
|
|
|
|
{
|
|
|
|
Method originalMethod = class_getInstanceMethod(cls, original);
|
|
|
|
IMP originalImplementation = method_getImplementation(originalMethod);
|
|
|
|
const char *originalArgTypes = method_getTypeEncoding(originalMethod);
|
2015-03-10 19:11:28 -07:00
|
|
|
|
2015-01-29 17:10:49 -08:00
|
|
|
Method replacementMethod = class_getInstanceMethod(cls, replacement);
|
|
|
|
IMP replacementImplementation = method_getImplementation(replacementMethod);
|
|
|
|
const char *replacementArgTypes = method_getTypeEncoding(replacementMethod);
|
2015-03-10 19:11:28 -07:00
|
|
|
|
2015-01-29 17:10:49 -08:00
|
|
|
if (class_addMethod(cls, original, replacementImplementation, replacementArgTypes))
|
|
|
|
{
|
|
|
|
class_replaceMethod(cls, replacement, originalImplementation, originalArgTypes);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
method_exchangeImplementations(originalMethod, replacementMethod);
|
|
|
|
}
|
2015-02-03 16:15:20 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
BOOL RCTClassOverridesClassMethod(Class cls, SEL selector)
|
|
|
|
{
|
|
|
|
return RCTClassOverridesInstanceMethod(object_getClass(cls), selector);
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOL RCTClassOverridesInstanceMethod(Class cls, SEL selector)
|
|
|
|
{
|
|
|
|
unsigned int numberOfMethods;
|
|
|
|
Method *methods = class_copyMethodList(cls, &numberOfMethods);
|
|
|
|
for (unsigned int i = 0; i < numberOfMethods; i++)
|
|
|
|
{
|
|
|
|
if (method_getName(methods[i]) == selector)
|
|
|
|
{
|
|
|
|
free(methods);
|
|
|
|
return YES;
|
|
|
|
}
|
|
|
|
}
|
2015-03-02 11:36:55 -08:00
|
|
|
free(methods);
|
2015-02-03 16:15:20 -08:00
|
|
|
return NO;
|
|
|
|
}
|
|
|
|
|
2015-03-02 11:36:55 -08:00
|
|
|
void RCTEnumerateClasses(void (^block)(Class cls))
|
|
|
|
{
|
|
|
|
static Class *classes;
|
|
|
|
static unsigned int classCount;
|
|
|
|
static dispatch_once_t onceToken;
|
|
|
|
dispatch_once(&onceToken, ^{
|
|
|
|
classes = objc_copyClassList(&classCount);
|
|
|
|
});
|
|
|
|
|
|
|
|
for (unsigned int i = 0; i < classCount; i++)
|
|
|
|
{
|
|
|
|
Class cls = classes[i];
|
|
|
|
Class superclass = cls;
|
|
|
|
while (superclass)
|
|
|
|
{
|
|
|
|
if (class_conformsToProtocol(superclass, @protocol(NSObject)))
|
|
|
|
{
|
|
|
|
block(cls);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
superclass = class_getSuperclass(superclass);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2015-03-12 12:51:44 -07:00
|
|
|
|
|
|
|
NSDictionary *RCTMakeError(NSString *message, id toStringify, NSDictionary *extraData)
|
|
|
|
{
|
|
|
|
if (toStringify) {
|
|
|
|
message = [NSString stringWithFormat:@"%@%@", message, toStringify];
|
|
|
|
}
|
|
|
|
NSMutableDictionary *error = [@{@"message": message} mutableCopy];
|
|
|
|
if (extraData) {
|
|
|
|
[error addEntriesFromDictionary:extraData];
|
|
|
|
}
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
|
|
|
NSDictionary *RCTMakeAndLogError(NSString *message, id toStringify, NSDictionary *extraData)
|
|
|
|
{
|
|
|
|
id error = RCTMakeError(message, toStringify, extraData);
|
|
|
|
RCTLogError(@"\nError: %@", error);
|
|
|
|
return error;
|
|
|
|
}
|