/** * 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 "RCTUtils.h" #import #import #import #import #import "RCTLog.h" 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) { if (!jsonString) { return nil; } NSData *jsonData = [jsonString dataUsingEncoding:NSUTF8StringEncoding allowLossyConversion:NO]; if (!jsonData) { jsonData = [jsonString dataUsingEncoding:NSUTF8StringEncoding allowLossyConversion:YES]; 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; } } 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() { static CGFloat scale; 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; } 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; } }); return size; } 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); }); 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); Method replacementMethod = class_getClassMethod(cls, replacement); IMP replacementImplementation = method_getImplementation(replacementMethod); const char *replacementArgTypes = method_getTypeEncoding(replacementMethod); 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); Method replacementMethod = class_getInstanceMethod(cls, replacement); IMP replacementImplementation = method_getImplementation(replacementMethod); const char *replacementArgTypes = method_getTypeEncoding(replacementMethod); if (class_addMethod(cls, original, replacementImplementation, replacementArgTypes)) { class_replaceMethod(cls, replacement, originalImplementation, originalArgTypes); } else { method_exchangeImplementations(originalMethod, replacementMethod); } } 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; } } free(methods); return NO; } 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); } } } 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; }