├── Example ├── Podfile ├── Podfile.lock ├── Pods │ ├── FBSnapshotTestCase │ │ ├── FBSnapshotTestCase │ │ │ ├── Categories │ │ │ │ ├── UIApplication+StrictKeyWindow.h │ │ │ │ ├── UIApplication+StrictKeyWindow.m │ │ │ │ ├── UIImage+Compare.h │ │ │ │ ├── UIImage+Compare.m │ │ │ │ ├── UIImage+Diff.h │ │ │ │ ├── UIImage+Diff.m │ │ │ │ ├── UIImage+Snapshot.h │ │ │ │ └── UIImage+Snapshot.m │ │ │ ├── FBSnapshotTestCase.h │ │ │ ├── FBSnapshotTestCase.m │ │ │ ├── FBSnapshotTestCasePlatform.h │ │ │ ├── FBSnapshotTestCasePlatform.m │ │ │ ├── FBSnapshotTestController.h │ │ │ ├── FBSnapshotTestController.m │ │ │ └── SwiftSupport.swift │ │ ├── LICENSE │ │ └── README.md │ ├── Local Podspecs │ │ └── ZLBridge.podspec.json │ ├── Manifest.lock │ ├── Pods.xcodeproj │ │ ├── project.pbxproj │ │ └── xcuserdata │ │ │ └── fanpeng.xcuserdatad │ │ │ └── xcschemes │ │ │ ├── FBSnapshotTestCase.xcscheme │ │ │ ├── Pods-ZLBridge_Example.xcscheme │ │ │ ├── Pods-ZLBridge_Tests.xcscheme │ │ │ ├── ZLBridge-ZLBridge.xcscheme │ │ │ ├── ZLBridge.xcscheme │ │ │ └── xcschememanagement.plist │ └── Target Support Files │ │ ├── FBSnapshotTestCase │ │ ├── FBSnapshotTestCase-Info.plist │ │ ├── FBSnapshotTestCase-dummy.m │ │ ├── FBSnapshotTestCase-prefix.pch │ │ ├── FBSnapshotTestCase-umbrella.h │ │ ├── FBSnapshotTestCase.debug.xcconfig │ │ ├── FBSnapshotTestCase.modulemap │ │ └── FBSnapshotTestCase.release.xcconfig │ │ ├── Pods-ZLBridge_Example │ │ ├── Pods-ZLBridge_Example-Info.plist │ │ ├── Pods-ZLBridge_Example-acknowledgements.markdown │ │ ├── Pods-ZLBridge_Example-acknowledgements.plist │ │ ├── Pods-ZLBridge_Example-dummy.m │ │ ├── Pods-ZLBridge_Example-frameworks.sh │ │ ├── Pods-ZLBridge_Example-umbrella.h │ │ ├── Pods-ZLBridge_Example.debug.xcconfig │ │ ├── Pods-ZLBridge_Example.modulemap │ │ └── Pods-ZLBridge_Example.release.xcconfig │ │ ├── Pods-ZLBridge_Tests │ │ ├── Pods-ZLBridge_Tests-Info.plist │ │ ├── Pods-ZLBridge_Tests-acknowledgements.markdown │ │ ├── Pods-ZLBridge_Tests-acknowledgements.plist │ │ ├── Pods-ZLBridge_Tests-dummy.m │ │ ├── Pods-ZLBridge_Tests-frameworks.sh │ │ ├── Pods-ZLBridge_Tests-umbrella.h │ │ ├── Pods-ZLBridge_Tests.debug.xcconfig │ │ ├── Pods-ZLBridge_Tests.modulemap │ │ └── Pods-ZLBridge_Tests.release.xcconfig │ │ └── ZLBridge │ │ ├── ResourceBundle-ZLBridge-ZLBridge-Info.plist │ │ ├── ZLBridge-Info.plist │ │ ├── ZLBridge-dummy.m │ │ ├── ZLBridge-prefix.pch │ │ ├── ZLBridge-umbrella.h │ │ ├── ZLBridge.debug.xcconfig │ │ ├── ZLBridge.modulemap │ │ └── ZLBridge.release.xcconfig ├── Tests │ ├── Tests-Info.plist │ ├── Tests-Prefix.pch │ ├── Tests.m │ └── en.lproj │ │ └── InfoPlist.strings ├── ZLBridge.xcodeproj │ ├── project.pbxproj │ ├── project.xcworkspace │ │ └── contents.xcworkspacedata │ └── xcshareddata │ │ └── xcschemes │ │ └── ZLBridge-Example.xcscheme ├── ZLBridge.xcworkspace │ ├── contents.xcworkspacedata │ └── xcuserdata │ │ └── fanpeng.xcuserdatad │ │ ├── UserInterfaceState.xcuserstate │ │ └── xcdebugger │ │ └── Breakpoints_v2.xcbkptlist └── ZLBridge │ ├── Base.lproj │ ├── LaunchScreen.storyboard │ └── Main.storyboard │ ├── Images.xcassets │ └── AppIcon.appiconset │ │ └── Contents.json │ ├── ZLAppDelegate.h │ ├── ZLAppDelegate.m │ ├── ZLBridge-Info.plist │ ├── ZLBridge-Prefix.pch │ ├── ZLTextVC.h │ ├── ZLTextVC.m │ ├── ZLTextVC.xib │ ├── ZLViewController.h │ ├── ZLViewController.m │ ├── ZLWebview.h │ ├── ZLWebview.m │ ├── en.lproj │ └── InfoPlist.strings │ ├── index.html │ └── main.m ├── LICENSE ├── README.md ├── ZLBridge.podspec ├── ZLBridge ├── Assets │ ├── .gitkeep │ └── ZLBridge.js └── Classes │ ├── .gitkeep │ ├── WKWebView+ZLBridge.h │ └── WKWebView+ZLBridge.m └── _Pods.xcodeproj /Example/Podfile: -------------------------------------------------------------------------------- 1 | use_frameworks! 2 | 3 | platform :ios, '9.0' 4 | 5 | target 'ZLBridge_Example' do 6 | pod 'ZLBridge', :path => '../' 7 | 8 | target 'ZLBridge_Tests' do 9 | inherit! :search_paths 10 | 11 | pod 'FBSnapshotTestCase' 12 | end 13 | # pod trunk push ZLBridge.podspec --allow-warnings --verbose 14 | 15 | end 16 | -------------------------------------------------------------------------------- /Example/Podfile.lock: -------------------------------------------------------------------------------- 1 | PODS: 2 | - FBSnapshotTestCase (2.1.4): 3 | - FBSnapshotTestCase/SwiftSupport (= 2.1.4) 4 | - FBSnapshotTestCase/Core (2.1.4) 5 | - FBSnapshotTestCase/SwiftSupport (2.1.4): 6 | - FBSnapshotTestCase/Core 7 | - ZLBridge (1.0.0) 8 | 9 | DEPENDENCIES: 10 | - FBSnapshotTestCase 11 | - ZLBridge (from `../`) 12 | 13 | SPEC REPOS: 14 | trunk: 15 | - FBSnapshotTestCase 16 | 17 | EXTERNAL SOURCES: 18 | ZLBridge: 19 | :path: "../" 20 | 21 | SPEC CHECKSUMS: 22 | FBSnapshotTestCase: 094f9f314decbabe373b87cc339bea235a63e07a 23 | ZLBridge: 5202001bbf512ad35072d1c993f55ffff9247a6b 24 | 25 | PODFILE CHECKSUM: affccafbf7f67f785a478373a0839b65dfda2bc8 26 | 27 | COCOAPODS: 1.9.3 28 | -------------------------------------------------------------------------------- /Example/Pods/FBSnapshotTestCase/FBSnapshotTestCase/Categories/UIApplication+StrictKeyWindow.h: -------------------------------------------------------------------------------- 1 | /* 2 | * Copyright (c) 2015, Facebook, Inc. 3 | * All rights reserved. 4 | * 5 | * This source code is licensed under the BSD-style license found in the 6 | * LICENSE file in the root directory of this source tree. An additional grant 7 | * of patent rights can be found in the PATENTS file in the same directory. 8 | * 9 | */ 10 | 11 | #import 12 | 13 | @interface UIApplication (StrictKeyWindow) 14 | 15 | /** 16 | @return The receiver's @c keyWindow. Raises an assertion if @c nil. 17 | */ 18 | - (UIWindow *)fb_strictKeyWindow; 19 | 20 | @end 21 | -------------------------------------------------------------------------------- /Example/Pods/FBSnapshotTestCase/FBSnapshotTestCase/Categories/UIApplication+StrictKeyWindow.m: -------------------------------------------------------------------------------- 1 | /* 2 | * Copyright (c) 2015, Facebook, Inc. 3 | * All rights reserved. 4 | * 5 | * This source code is licensed under the BSD-style license found in the 6 | * LICENSE file in the root directory of this source tree. An additional grant 7 | * of patent rights can be found in the PATENTS file in the same directory. 8 | * 9 | */ 10 | 11 | #import 12 | 13 | @implementation UIApplication (StrictKeyWindow) 14 | 15 | - (UIWindow *)fb_strictKeyWindow 16 | { 17 | UIWindow *keyWindow = [UIApplication sharedApplication].keyWindow; 18 | if (!keyWindow) { 19 | [NSException raise:@"FBSnapshotTestCaseNilKeyWindowException" 20 | format:@"Snapshot tests must be hosted by an application with a key window. Please ensure your test" 21 | " host sets up a key window at launch (either via storyboards or programmatically) and doesn't" 22 | " do anything to remove it while snapshot tests are running."]; 23 | } 24 | return keyWindow; 25 | } 26 | 27 | @end 28 | -------------------------------------------------------------------------------- /Example/Pods/FBSnapshotTestCase/FBSnapshotTestCase/Categories/UIImage+Compare.h: -------------------------------------------------------------------------------- 1 | // 2 | // Created by Gabriel Handford on 3/1/09. 3 | // Copyright 2009-2013. All rights reserved. 4 | // Created by John Boiles on 10/20/11. 5 | // Copyright (c) 2011. All rights reserved 6 | // Modified by Felix Schulze on 2/11/13. 7 | // Copyright 2013. All rights reserved. 8 | // 9 | // Permission is hereby granted, free of charge, to any person 10 | // obtaining a copy of this software and associated documentation 11 | // files (the "Software"), to deal in the Software without 12 | // restriction, including without limitation the rights to use, 13 | // copy, modify, merge, publish, distribute, sublicense, and/or sell 14 | // copies of the Software, and to permit persons to whom the 15 | // Software is furnished to do so, subject to the following 16 | // conditions: 17 | // 18 | // The above copyright notice and this permission notice shall be 19 | // included in all copies or substantial portions of the Software. 20 | // 21 | // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, 22 | // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES 23 | // OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND 24 | // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT 25 | // HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, 26 | // WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING 27 | // FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR 28 | // OTHER DEALINGS IN THE SOFTWARE. 29 | // 30 | 31 | #import 32 | 33 | @interface UIImage (Compare) 34 | 35 | - (BOOL)fb_compareWithImage:(UIImage *)image tolerance:(CGFloat)tolerance; 36 | 37 | @end 38 | -------------------------------------------------------------------------------- /Example/Pods/FBSnapshotTestCase/FBSnapshotTestCase/Categories/UIImage+Compare.m: -------------------------------------------------------------------------------- 1 | // 2 | // Created by Gabriel Handford on 3/1/09. 3 | // Copyright 2009-2013. All rights reserved. 4 | // Created by John Boiles on 10/20/11. 5 | // Copyright (c) 2011. All rights reserved 6 | // Modified by Felix Schulze on 2/11/13. 7 | // Copyright 2013. All rights reserved. 8 | // 9 | // Permission is hereby granted, free of charge, to any person 10 | // obtaining a copy of this software and associated documentation 11 | // files (the "Software"), to deal in the Software without 12 | // restriction, including without limitation the rights to use, 13 | // copy, modify, merge, publish, distribute, sublicense, and/or sell 14 | // copies of the Software, and to permit persons to whom the 15 | // Software is furnished to do so, subject to the following 16 | // conditions: 17 | // 18 | // The above copyright notice and this permission notice shall be 19 | // included in all copies or substantial portions of the Software. 20 | // 21 | // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, 22 | // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES 23 | // OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND 24 | // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT 25 | // HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, 26 | // WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING 27 | // FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR 28 | // OTHER DEALINGS IN THE SOFTWARE. 29 | // 30 | 31 | #import 32 | 33 | // This makes debugging much more fun 34 | typedef union { 35 | uint32_t raw; 36 | unsigned char bytes[4]; 37 | struct { 38 | char red; 39 | char green; 40 | char blue; 41 | char alpha; 42 | } __attribute__ ((packed)) pixels; 43 | } FBComparePixel; 44 | 45 | @implementation UIImage (Compare) 46 | 47 | - (BOOL)fb_compareWithImage:(UIImage *)image tolerance:(CGFloat)tolerance 48 | { 49 | NSAssert(CGSizeEqualToSize(self.size, image.size), @"Images must be same size."); 50 | 51 | CGSize referenceImageSize = CGSizeMake(CGImageGetWidth(self.CGImage), CGImageGetHeight(self.CGImage)); 52 | CGSize imageSize = CGSizeMake(CGImageGetWidth(image.CGImage), CGImageGetHeight(image.CGImage)); 53 | 54 | // The images have the equal size, so we could use the smallest amount of bytes because of byte padding 55 | size_t minBytesPerRow = MIN(CGImageGetBytesPerRow(self.CGImage), CGImageGetBytesPerRow(image.CGImage)); 56 | size_t referenceImageSizeBytes = referenceImageSize.height * minBytesPerRow; 57 | void *referenceImagePixels = calloc(1, referenceImageSizeBytes); 58 | void *imagePixels = calloc(1, referenceImageSizeBytes); 59 | 60 | if (!referenceImagePixels || !imagePixels) { 61 | free(referenceImagePixels); 62 | free(imagePixels); 63 | return NO; 64 | } 65 | 66 | CGContextRef referenceImageContext = CGBitmapContextCreate(referenceImagePixels, 67 | referenceImageSize.width, 68 | referenceImageSize.height, 69 | CGImageGetBitsPerComponent(self.CGImage), 70 | minBytesPerRow, 71 | CGImageGetColorSpace(self.CGImage), 72 | (CGBitmapInfo)kCGImageAlphaPremultipliedLast 73 | ); 74 | CGContextRef imageContext = CGBitmapContextCreate(imagePixels, 75 | imageSize.width, 76 | imageSize.height, 77 | CGImageGetBitsPerComponent(image.CGImage), 78 | minBytesPerRow, 79 | CGImageGetColorSpace(image.CGImage), 80 | (CGBitmapInfo)kCGImageAlphaPremultipliedLast 81 | ); 82 | 83 | if (!referenceImageContext || !imageContext) { 84 | CGContextRelease(referenceImageContext); 85 | CGContextRelease(imageContext); 86 | free(referenceImagePixels); 87 | free(imagePixels); 88 | return NO; 89 | } 90 | 91 | CGContextDrawImage(referenceImageContext, CGRectMake(0, 0, referenceImageSize.width, referenceImageSize.height), self.CGImage); 92 | CGContextDrawImage(imageContext, CGRectMake(0, 0, imageSize.width, imageSize.height), image.CGImage); 93 | 94 | CGContextRelease(referenceImageContext); 95 | CGContextRelease(imageContext); 96 | 97 | BOOL imageEqual = YES; 98 | 99 | // Do a fast compare if we can 100 | if (tolerance == 0) { 101 | imageEqual = (memcmp(referenceImagePixels, imagePixels, referenceImageSizeBytes) == 0); 102 | } else { 103 | // Go through each pixel in turn and see if it is different 104 | const NSInteger pixelCount = referenceImageSize.width * referenceImageSize.height; 105 | 106 | FBComparePixel *p1 = referenceImagePixels; 107 | FBComparePixel *p2 = imagePixels; 108 | 109 | NSInteger numDiffPixels = 0; 110 | for (int n = 0; n < pixelCount; ++n) { 111 | // If this pixel is different, increment the pixel diff count and see 112 | // if we have hit our limit. 113 | if (p1->raw != p2->raw) { 114 | numDiffPixels ++; 115 | 116 | CGFloat percent = (CGFloat)numDiffPixels / pixelCount; 117 | if (percent > tolerance) { 118 | imageEqual = NO; 119 | break; 120 | } 121 | } 122 | 123 | p1++; 124 | p2++; 125 | } 126 | } 127 | 128 | free(referenceImagePixels); 129 | free(imagePixels); 130 | 131 | return imageEqual; 132 | } 133 | 134 | @end 135 | -------------------------------------------------------------------------------- /Example/Pods/FBSnapshotTestCase/FBSnapshotTestCase/Categories/UIImage+Diff.h: -------------------------------------------------------------------------------- 1 | // 2 | // Created by Gabriel Handford on 3/1/09. 3 | // Copyright 2009-2013. All rights reserved. 4 | // Created by John Boiles on 10/20/11. 5 | // Copyright (c) 2011. All rights reserved 6 | // Modified by Felix Schulze on 2/11/13. 7 | // Copyright 2013. All rights reserved. 8 | // 9 | // Permission is hereby granted, free of charge, to any person 10 | // obtaining a copy of this software and associated documentation 11 | // files (the "Software"), to deal in the Software without 12 | // restriction, including without limitation the rights to use, 13 | // copy, modify, merge, publish, distribute, sublicense, and/or sell 14 | // copies of the Software, and to permit persons to whom the 15 | // Software is furnished to do so, subject to the following 16 | // conditions: 17 | // 18 | // The above copyright notice and this permission notice shall be 19 | // included in all copies or substantial portions of the Software. 20 | // 21 | // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, 22 | // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES 23 | // OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND 24 | // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT 25 | // HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, 26 | // WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING 27 | // FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR 28 | // OTHER DEALINGS IN THE SOFTWARE. 29 | // 30 | 31 | #import 32 | 33 | @interface UIImage (Diff) 34 | 35 | - (UIImage *)fb_diffWithImage:(UIImage *)image; 36 | 37 | @end 38 | -------------------------------------------------------------------------------- /Example/Pods/FBSnapshotTestCase/FBSnapshotTestCase/Categories/UIImage+Diff.m: -------------------------------------------------------------------------------- 1 | // 2 | // Created by Gabriel Handford on 3/1/09. 3 | // Copyright 2009-2013. All rights reserved. 4 | // Created by John Boiles on 10/20/11. 5 | // Copyright (c) 2011. All rights reserved 6 | // Modified by Felix Schulze on 2/11/13. 7 | // Copyright 2013. All rights reserved. 8 | // 9 | // Permission is hereby granted, free of charge, to any person 10 | // obtaining a copy of this software and associated documentation 11 | // files (the "Software"), to deal in the Software without 12 | // restriction, including without limitation the rights to use, 13 | // copy, modify, merge, publish, distribute, sublicense, and/or sell 14 | // copies of the Software, and to permit persons to whom the 15 | // Software is furnished to do so, subject to the following 16 | // conditions: 17 | // 18 | // The above copyright notice and this permission notice shall be 19 | // included in all copies or substantial portions of the Software. 20 | // 21 | // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, 22 | // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES 23 | // OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND 24 | // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT 25 | // HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, 26 | // WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING 27 | // FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR 28 | // OTHER DEALINGS IN THE SOFTWARE. 29 | // 30 | 31 | #import 32 | 33 | @implementation UIImage (Diff) 34 | 35 | - (UIImage *)fb_diffWithImage:(UIImage *)image 36 | { 37 | if (!image) { 38 | return nil; 39 | } 40 | CGSize imageSize = CGSizeMake(MAX(self.size.width, image.size.width), MAX(self.size.height, image.size.height)); 41 | UIGraphicsBeginImageContextWithOptions(imageSize, YES, 0); 42 | CGContextRef context = UIGraphicsGetCurrentContext(); 43 | [self drawInRect:CGRectMake(0, 0, self.size.width, self.size.height)]; 44 | CGContextSetAlpha(context, 0.5); 45 | CGContextBeginTransparencyLayer(context, NULL); 46 | [image drawInRect:CGRectMake(0, 0, image.size.width, image.size.height)]; 47 | CGContextSetBlendMode(context, kCGBlendModeDifference); 48 | CGContextSetFillColorWithColor(context,[UIColor whiteColor].CGColor); 49 | CGContextFillRect(context, CGRectMake(0, 0, self.size.width, self.size.height)); 50 | CGContextEndTransparencyLayer(context); 51 | UIImage *returnImage = UIGraphicsGetImageFromCurrentImageContext(); 52 | UIGraphicsEndImageContext(); 53 | return returnImage; 54 | } 55 | 56 | @end 57 | -------------------------------------------------------------------------------- /Example/Pods/FBSnapshotTestCase/FBSnapshotTestCase/Categories/UIImage+Snapshot.h: -------------------------------------------------------------------------------- 1 | /* 2 | * Copyright (c) 2015, Facebook, Inc. 3 | * All rights reserved. 4 | * 5 | * This source code is licensed under the BSD-style license found in the 6 | * LICENSE file in the root directory of this source tree. An additional grant 7 | * of patent rights can be found in the PATENTS file in the same directory. 8 | * 9 | */ 10 | 11 | #import 12 | 13 | @interface UIImage (Snapshot) 14 | 15 | /// Uses renderInContext: to get a snapshot of the layer. 16 | + (UIImage *)fb_imageForLayer:(CALayer *)layer; 17 | 18 | /// Uses renderInContext: to get a snapshot of the view layer. 19 | + (UIImage *)fb_imageForViewLayer:(UIView *)view; 20 | 21 | /// Uses drawViewHierarchyInRect: to get a snapshot of the view and adds the view into a window if needed. 22 | + (UIImage *)fb_imageForView:(UIView *)view; 23 | 24 | @end 25 | -------------------------------------------------------------------------------- /Example/Pods/FBSnapshotTestCase/FBSnapshotTestCase/Categories/UIImage+Snapshot.m: -------------------------------------------------------------------------------- 1 | /* 2 | * Copyright (c) 2015, Facebook, Inc. 3 | * All rights reserved. 4 | * 5 | * This source code is licensed under the BSD-style license found in the 6 | * LICENSE file in the root directory of this source tree. An additional grant 7 | * of patent rights can be found in the PATENTS file in the same directory. 8 | * 9 | */ 10 | 11 | #import 12 | #import 13 | 14 | @implementation UIImage (Snapshot) 15 | 16 | + (UIImage *)fb_imageForLayer:(CALayer *)layer 17 | { 18 | CGRect bounds = layer.bounds; 19 | NSAssert1(CGRectGetWidth(bounds), @"Zero width for layer %@", layer); 20 | NSAssert1(CGRectGetHeight(bounds), @"Zero height for layer %@", layer); 21 | 22 | UIGraphicsBeginImageContextWithOptions(bounds.size, NO, 0); 23 | CGContextRef context = UIGraphicsGetCurrentContext(); 24 | NSAssert1(context, @"Could not generate context for layer %@", layer); 25 | CGContextSaveGState(context); 26 | [layer layoutIfNeeded]; 27 | [layer renderInContext:context]; 28 | CGContextRestoreGState(context); 29 | 30 | UIImage *snapshot = UIGraphicsGetImageFromCurrentImageContext(); 31 | UIGraphicsEndImageContext(); 32 | return snapshot; 33 | } 34 | 35 | + (UIImage *)fb_imageForViewLayer:(UIView *)view 36 | { 37 | [view layoutIfNeeded]; 38 | return [self fb_imageForLayer:view.layer]; 39 | } 40 | 41 | + (UIImage *)fb_imageForView:(UIView *)view 42 | { 43 | CGRect bounds = view.bounds; 44 | NSAssert1(CGRectGetWidth(bounds), @"Zero width for view %@", view); 45 | NSAssert1(CGRectGetHeight(bounds), @"Zero height for view %@", view); 46 | 47 | // If the input view is already a UIWindow, then just use that. Otherwise wrap in a window. 48 | UIWindow *window = [view isKindOfClass:[UIWindow class]] ? (UIWindow *)view : view.window; 49 | BOOL removeFromSuperview = NO; 50 | if (!window) { 51 | window = [[UIApplication sharedApplication] fb_strictKeyWindow]; 52 | } 53 | 54 | if (!view.window && view != window) { 55 | [window addSubview:view]; 56 | removeFromSuperview = YES; 57 | } 58 | 59 | UIGraphicsBeginImageContextWithOptions(bounds.size, NO, 0); 60 | [view layoutIfNeeded]; 61 | [view drawViewHierarchyInRect:view.bounds afterScreenUpdates:YES]; 62 | 63 | UIImage *snapshot = UIGraphicsGetImageFromCurrentImageContext(); 64 | UIGraphicsEndImageContext(); 65 | 66 | if (removeFromSuperview) { 67 | [view removeFromSuperview]; 68 | } 69 | 70 | return snapshot; 71 | } 72 | 73 | @end 74 | -------------------------------------------------------------------------------- /Example/Pods/FBSnapshotTestCase/FBSnapshotTestCase/FBSnapshotTestCase.h: -------------------------------------------------------------------------------- 1 | /* 2 | * Copyright (c) 2015, Facebook, Inc. 3 | * All rights reserved. 4 | * 5 | * This source code is licensed under the BSD-style license found in the 6 | * LICENSE file in the root directory of this source tree. An additional grant 7 | * of patent rights can be found in the PATENTS file in the same directory. 8 | * 9 | */ 10 | 11 | #import 12 | #import 13 | 14 | #import 15 | 16 | #import 17 | 18 | #import 19 | 20 | /* 21 | There are three ways of setting reference image directories. 22 | 23 | 1. Set the preprocessor macro FB_REFERENCE_IMAGE_DIR to a double quoted 24 | c-string with the path. 25 | 2. Set an environment variable named FB_REFERENCE_IMAGE_DIR with the path. This 26 | takes precedence over the preprocessor macro to allow for run-time override. 27 | 3. Keep everything unset, which will cause the reference images to be looked up 28 | inside the bundle holding the current test, in the 29 | Resources/ReferenceImages_* directories. 30 | */ 31 | #ifndef FB_REFERENCE_IMAGE_DIR 32 | #define FB_REFERENCE_IMAGE_DIR "" 33 | #endif 34 | 35 | /** 36 | Similar to our much-loved XCTAssert() macros. Use this to perform your test. No need to write an explanation, though. 37 | @param view The view to snapshot 38 | @param identifier An optional identifier, used if there are multiple snapshot tests in a given -test method. 39 | @param suffixes An NSOrderedSet of strings for the different suffixes 40 | @param tolerance The percentage of pixels that can differ and still count as an 'identical' view 41 | */ 42 | #define FBSnapshotVerifyViewWithOptions(view__, identifier__, suffixes__, tolerance__) \ 43 | FBSnapshotVerifyViewOrLayerWithOptions(View, view__, identifier__, suffixes__, tolerance__) 44 | 45 | #define FBSnapshotVerifyView(view__, identifier__) \ 46 | FBSnapshotVerifyViewWithOptions(view__, identifier__, FBSnapshotTestCaseDefaultSuffixes(), 0) 47 | 48 | 49 | /** 50 | Similar to our much-loved XCTAssert() macros. Use this to perform your test. No need to write an explanation, though. 51 | @param layer The layer to snapshot 52 | @param identifier An optional identifier, used if there are multiple snapshot tests in a given -test method. 53 | @param suffixes An NSOrderedSet of strings for the different suffixes 54 | @param tolerance The percentage of pixels that can differ and still count as an 'identical' layer 55 | */ 56 | #define FBSnapshotVerifyLayerWithOptions(layer__, identifier__, suffixes__, tolerance__) \ 57 | FBSnapshotVerifyViewOrLayerWithOptions(Layer, layer__, identifier__, suffixes__, tolerance__) 58 | 59 | #define FBSnapshotVerifyLayer(layer__, identifier__) \ 60 | FBSnapshotVerifyLayerWithOptions(layer__, identifier__, FBSnapshotTestCaseDefaultSuffixes(), 0) 61 | 62 | 63 | #define FBSnapshotVerifyViewOrLayerWithOptions(what__, viewOrLayer__, identifier__, suffixes__, tolerance__) \ 64 | { \ 65 | NSString *errorDescription = [self snapshotVerifyViewOrLayer:viewOrLayer__ identifier:identifier__ suffixes:suffixes__ tolerance:tolerance__]; \ 66 | BOOL noErrors = (errorDescription == nil); \ 67 | XCTAssertTrue(noErrors, @"%@", errorDescription); \ 68 | } 69 | 70 | 71 | /** 72 | The base class of view snapshotting tests. If you have small UI component, it's often easier to configure it in a test 73 | and compare an image of the view to a reference image that write lots of complex layout-code tests. 74 | 75 | In order to flip the tests in your subclass to record the reference images set @c recordMode to @c YES. 76 | 77 | @attention When recording, the reference image directory should be explicitly 78 | set, otherwise the images may be written to somewhere inside the 79 | simulator directory. 80 | 81 | For example: 82 | @code 83 | - (void)setUp 84 | { 85 | [super setUp]; 86 | self.recordMode = YES; 87 | } 88 | @endcode 89 | */ 90 | @interface FBSnapshotTestCase : XCTestCase 91 | 92 | /** 93 | When YES, the test macros will save reference images, rather than performing an actual test. 94 | */ 95 | @property (readwrite, nonatomic, assign) BOOL recordMode; 96 | 97 | /** 98 | When @c YES appends the name of the device model and OS to the snapshot file name. 99 | The default value is @c NO. 100 | */ 101 | @property (readwrite, nonatomic, assign, getter=isDeviceAgnostic) BOOL deviceAgnostic; 102 | 103 | /** 104 | When YES, renders a snapshot of the complete view hierarchy as visible onscreen. 105 | There are several things that do not work if renderInContext: is used. 106 | - UIVisualEffect #70 107 | - UIAppearance #91 108 | - Size Classes #92 109 | 110 | @attention If the view does't belong to a UIWindow, it will create one and add the view as a subview. 111 | */ 112 | @property (readwrite, nonatomic, assign) BOOL usesDrawViewHierarchyInRect; 113 | 114 | - (void)setUp NS_REQUIRES_SUPER; 115 | - (void)tearDown NS_REQUIRES_SUPER; 116 | 117 | /** 118 | Performs the comparison or records a snapshot of the layer if recordMode is YES. 119 | @param viewOrLayer The UIView or CALayer to snapshot 120 | @param identifier An optional identifier, used if there are multiple snapshot tests in a given -test method. 121 | @param suffixes An NSOrderedSet of strings for the different suffixes 122 | @param tolerance The percentage difference to still count as identical - 0 mean pixel perfect, 1 means I don't care 123 | @returns nil if the comparison (or saving of the reference image) succeeded. Otherwise it contains an error description. 124 | */ 125 | - (NSString *)snapshotVerifyViewOrLayer:(id)viewOrLayer 126 | identifier:(NSString *)identifier 127 | suffixes:(NSOrderedSet *)suffixes 128 | tolerance:(CGFloat)tolerance; 129 | 130 | /** 131 | Performs the comparison or records a snapshot of the layer if recordMode is YES. 132 | @param layer The Layer to snapshot 133 | @param referenceImagesDirectory The directory in which reference images are stored. 134 | @param identifier An optional identifier, used if there are multiple snapshot tests in a given -test method. 135 | @param tolerance The percentage difference to still count as identical - 0 mean pixel perfect, 1 means I don't care 136 | @param errorPtr An error to log in an XCTAssert() macro if the method fails (missing reference image, images differ, etc). 137 | @returns YES if the comparison (or saving of the reference image) succeeded. 138 | */ 139 | - (BOOL)compareSnapshotOfLayer:(CALayer *)layer 140 | referenceImagesDirectory:(NSString *)referenceImagesDirectory 141 | identifier:(NSString *)identifier 142 | tolerance:(CGFloat)tolerance 143 | error:(NSError **)errorPtr; 144 | 145 | /** 146 | Performs the comparison or records a snapshot of the view if recordMode is YES. 147 | @param view The view to snapshot 148 | @param referenceImagesDirectory The directory in which reference images are stored. 149 | @param identifier An optional identifier, used if there are multiple snapshot tests in a given -test method. 150 | @param tolerance The percentage difference to still count as identical - 0 mean pixel perfect, 1 means I don't care 151 | @param errorPtr An error to log in an XCTAssert() macro if the method fails (missing reference image, images differ, etc). 152 | @returns YES if the comparison (or saving of the reference image) succeeded. 153 | */ 154 | - (BOOL)compareSnapshotOfView:(UIView *)view 155 | referenceImagesDirectory:(NSString *)referenceImagesDirectory 156 | identifier:(NSString *)identifier 157 | tolerance:(CGFloat)tolerance 158 | error:(NSError **)errorPtr; 159 | 160 | /** 161 | Checks if reference image with identifier based name exists in the reference images directory. 162 | @param referenceImagesDirectory The directory in which reference images are stored. 163 | @param identifier An optional identifier, used if there are multiple snapshot tests in a given -test method. 164 | @param errorPtr An error to log in an XCTAssert() macro if the method fails (missing reference image, images differ, etc). 165 | @returns YES if reference image exists. 166 | */ 167 | - (BOOL)referenceImageRecordedInDirectory:(NSString *)referenceImagesDirectory 168 | identifier:(NSString *)identifier 169 | error:(NSError **)errorPtr; 170 | 171 | /** 172 | Returns the reference image directory. 173 | 174 | Helper function used to implement the assert macros. 175 | 176 | @param dir directory to use if environment variable not specified. Ignored if null or empty. 177 | */ 178 | - (NSString *)getReferenceImageDirectoryWithDefault:(NSString *)dir; 179 | 180 | @end 181 | -------------------------------------------------------------------------------- /Example/Pods/FBSnapshotTestCase/FBSnapshotTestCase/FBSnapshotTestCase.m: -------------------------------------------------------------------------------- 1 | /* 2 | * Copyright (c) 2015, Facebook, Inc. 3 | * All rights reserved. 4 | * 5 | * This source code is licensed under the BSD-style license found in the 6 | * LICENSE file in the root directory of this source tree. An additional grant 7 | * of patent rights can be found in the PATENTS file in the same directory. 8 | * 9 | */ 10 | 11 | #import 12 | #import 13 | 14 | @implementation FBSnapshotTestCase 15 | { 16 | FBSnapshotTestController *_snapshotController; 17 | } 18 | 19 | #pragma mark - Overrides 20 | 21 | - (void)setUp 22 | { 23 | [super setUp]; 24 | _snapshotController = [[FBSnapshotTestController alloc] initWithTestName:NSStringFromClass([self class])]; 25 | } 26 | 27 | - (void)tearDown 28 | { 29 | _snapshotController = nil; 30 | [super tearDown]; 31 | } 32 | 33 | - (BOOL)recordMode 34 | { 35 | return _snapshotController.recordMode; 36 | } 37 | 38 | - (void)setRecordMode:(BOOL)recordMode 39 | { 40 | NSAssert1(_snapshotController, @"%s cannot be called before [super setUp]", __FUNCTION__); 41 | _snapshotController.recordMode = recordMode; 42 | } 43 | 44 | - (BOOL)isDeviceAgnostic 45 | { 46 | return _snapshotController.deviceAgnostic; 47 | } 48 | 49 | - (void)setDeviceAgnostic:(BOOL)deviceAgnostic 50 | { 51 | NSAssert1(_snapshotController, @"%s cannot be called before [super setUp]", __FUNCTION__); 52 | _snapshotController.deviceAgnostic = deviceAgnostic; 53 | } 54 | 55 | - (BOOL)usesDrawViewHierarchyInRect 56 | { 57 | return _snapshotController.usesDrawViewHierarchyInRect; 58 | } 59 | 60 | - (void)setUsesDrawViewHierarchyInRect:(BOOL)usesDrawViewHierarchyInRect 61 | { 62 | NSAssert1(_snapshotController, @"%s cannot be called before [super setUp]", __FUNCTION__); 63 | _snapshotController.usesDrawViewHierarchyInRect = usesDrawViewHierarchyInRect; 64 | } 65 | 66 | #pragma mark - Public API 67 | 68 | - (NSString *)snapshotVerifyViewOrLayer:(id)viewOrLayer 69 | identifier:(NSString *)identifier 70 | suffixes:(NSOrderedSet *)suffixes 71 | tolerance:(CGFloat)tolerance 72 | { 73 | if (nil == viewOrLayer) { 74 | return @"Object to be snapshotted must not be nil"; 75 | } 76 | NSString *referenceImageDirectory = [self getReferenceImageDirectoryWithDefault:(@ FB_REFERENCE_IMAGE_DIR)]; 77 | if (referenceImageDirectory == nil) { 78 | return @"Missing value for referenceImagesDirectory - Set FB_REFERENCE_IMAGE_DIR as Environment variable in your scheme."; 79 | } 80 | if (suffixes.count == 0) { 81 | return [NSString stringWithFormat:@"Suffixes set cannot be empty %@", suffixes]; 82 | } 83 | 84 | BOOL testSuccess = NO; 85 | NSError *error = nil; 86 | NSMutableArray *errors = [NSMutableArray array]; 87 | 88 | if (self.recordMode) { 89 | NSString *referenceImagesDirectory = [NSString stringWithFormat:@"%@%@", referenceImageDirectory, suffixes.firstObject]; 90 | BOOL referenceImageSaved = [self _compareSnapshotOfViewOrLayer:viewOrLayer referenceImagesDirectory:referenceImagesDirectory identifier:(identifier) tolerance:tolerance error:&error]; 91 | if (!referenceImageSaved) { 92 | [errors addObject:error]; 93 | } 94 | } else { 95 | for (NSString *suffix in suffixes) { 96 | NSString *referenceImagesDirectory = [NSString stringWithFormat:@"%@%@", referenceImageDirectory, suffix]; 97 | BOOL referenceImageAvailable = [self referenceImageRecordedInDirectory:referenceImagesDirectory identifier:(identifier) error:&error]; 98 | 99 | if (referenceImageAvailable) { 100 | BOOL comparisonSuccess = [self _compareSnapshotOfViewOrLayer:viewOrLayer referenceImagesDirectory:referenceImagesDirectory identifier:identifier tolerance:tolerance error:&error]; 101 | [errors removeAllObjects]; 102 | if (comparisonSuccess) { 103 | testSuccess = YES; 104 | break; 105 | } else { 106 | [errors addObject:error]; 107 | } 108 | } else { 109 | [errors addObject:error]; 110 | } 111 | } 112 | } 113 | 114 | if (!testSuccess) { 115 | return [NSString stringWithFormat:@"Snapshot comparison failed: %@", errors.firstObject]; 116 | } 117 | if (self.recordMode) { 118 | return @"Test ran in record mode. Reference image is now saved. Disable record mode to perform an actual snapshot comparison!"; 119 | } 120 | 121 | return nil; 122 | } 123 | 124 | - (BOOL)compareSnapshotOfLayer:(CALayer *)layer 125 | referenceImagesDirectory:(NSString *)referenceImagesDirectory 126 | identifier:(NSString *)identifier 127 | tolerance:(CGFloat)tolerance 128 | error:(NSError **)errorPtr 129 | { 130 | return [self _compareSnapshotOfViewOrLayer:layer 131 | referenceImagesDirectory:referenceImagesDirectory 132 | identifier:identifier 133 | tolerance:tolerance 134 | error:errorPtr]; 135 | } 136 | 137 | - (BOOL)compareSnapshotOfView:(UIView *)view 138 | referenceImagesDirectory:(NSString *)referenceImagesDirectory 139 | identifier:(NSString *)identifier 140 | tolerance:(CGFloat)tolerance 141 | error:(NSError **)errorPtr 142 | { 143 | return [self _compareSnapshotOfViewOrLayer:view 144 | referenceImagesDirectory:referenceImagesDirectory 145 | identifier:identifier 146 | tolerance:tolerance 147 | error:errorPtr]; 148 | } 149 | 150 | - (BOOL)referenceImageRecordedInDirectory:(NSString *)referenceImagesDirectory 151 | identifier:(NSString *)identifier 152 | error:(NSError **)errorPtr 153 | { 154 | NSAssert1(_snapshotController, @"%s cannot be called before [super setUp]", __FUNCTION__); 155 | _snapshotController.referenceImagesDirectory = referenceImagesDirectory; 156 | UIImage *referenceImage = [_snapshotController referenceImageForSelector:self.invocation.selector 157 | identifier:identifier 158 | error:errorPtr]; 159 | 160 | return (referenceImage != nil); 161 | } 162 | 163 | - (NSString *)getReferenceImageDirectoryWithDefault:(NSString *)dir 164 | { 165 | NSString *envReferenceImageDirectory = [NSProcessInfo processInfo].environment[@"FB_REFERENCE_IMAGE_DIR"]; 166 | if (envReferenceImageDirectory) { 167 | return envReferenceImageDirectory; 168 | } 169 | if (dir && dir.length > 0) { 170 | return dir; 171 | } 172 | return [[NSBundle bundleForClass:self.class].resourcePath stringByAppendingPathComponent:@"ReferenceImages"]; 173 | } 174 | 175 | 176 | #pragma mark - Private API 177 | 178 | - (BOOL)_compareSnapshotOfViewOrLayer:(id)viewOrLayer 179 | referenceImagesDirectory:(NSString *)referenceImagesDirectory 180 | identifier:(NSString *)identifier 181 | tolerance:(CGFloat)tolerance 182 | error:(NSError **)errorPtr 183 | { 184 | _snapshotController.referenceImagesDirectory = referenceImagesDirectory; 185 | return [_snapshotController compareSnapshotOfViewOrLayer:viewOrLayer 186 | selector:self.invocation.selector 187 | identifier:identifier 188 | tolerance:tolerance 189 | error:errorPtr]; 190 | } 191 | 192 | @end 193 | -------------------------------------------------------------------------------- /Example/Pods/FBSnapshotTestCase/FBSnapshotTestCase/FBSnapshotTestCasePlatform.h: -------------------------------------------------------------------------------- 1 | /* 2 | * Copyright (c) 2015, Facebook, Inc. 3 | * All rights reserved. 4 | * 5 | * This source code is licensed under the BSD-style license found in the 6 | * LICENSE file in the root directory of this source tree. An additional grant 7 | * of patent rights can be found in the PATENTS file in the same directory. 8 | * 9 | */ 10 | 11 | #import 12 | 13 | #ifdef __cplusplus 14 | extern "C" { 15 | #endif 16 | 17 | /** 18 | Returns a Boolean value that indicates whether the snapshot test is running in 64Bit. 19 | This method is a convenience for creating the suffixes set based on the architecture 20 | that the test is running. 21 | 22 | @returns @c YES if the test is running in 64bit, otherwise @c NO. 23 | */ 24 | BOOL FBSnapshotTestCaseIs64Bit(void); 25 | 26 | /** 27 | Returns a default set of strings that is used to append a suffix based on the architectures. 28 | @warning Do not modify this function, you can create your own and use it with @c FBSnapshotVerifyViewWithOptions() 29 | 30 | @returns An @c NSOrderedSet object containing strings that are appended to the reference images directory. 31 | */ 32 | NSOrderedSet *FBSnapshotTestCaseDefaultSuffixes(void); 33 | 34 | /** 35 | Returns a fully «normalized» file name. 36 | Strips punctuation and spaces and replaces them with @c _. Also appends the device model, running OS and screen size to the file name. 37 | 38 | @returns An @c NSString object containing the passed @c fileName with the device model, OS and screen size appended at the end. 39 | */ 40 | NSString *FBDeviceAgnosticNormalizedFileName(NSString *fileName); 41 | 42 | #ifdef __cplusplus 43 | } 44 | #endif 45 | -------------------------------------------------------------------------------- /Example/Pods/FBSnapshotTestCase/FBSnapshotTestCase/FBSnapshotTestCasePlatform.m: -------------------------------------------------------------------------------- 1 | /* 2 | * Copyright (c) 2015, Facebook, Inc. 3 | * All rights reserved. 4 | * 5 | * This source code is licensed under the BSD-style license found in the 6 | * LICENSE file in the root directory of this source tree. An additional grant 7 | * of patent rights can be found in the PATENTS file in the same directory. 8 | * 9 | */ 10 | 11 | #import 12 | #import 13 | #import 14 | 15 | BOOL FBSnapshotTestCaseIs64Bit(void) 16 | { 17 | #if __LP64__ 18 | return YES; 19 | #else 20 | return NO; 21 | #endif 22 | } 23 | 24 | NSOrderedSet *FBSnapshotTestCaseDefaultSuffixes(void) 25 | { 26 | NSMutableOrderedSet *suffixesSet = [[NSMutableOrderedSet alloc] init]; 27 | [suffixesSet addObject:@"_32"]; 28 | [suffixesSet addObject:@"_64"]; 29 | if (FBSnapshotTestCaseIs64Bit()) { 30 | return [suffixesSet reversedOrderedSet]; 31 | } 32 | return [suffixesSet copy]; 33 | } 34 | 35 | NSString *FBDeviceAgnosticNormalizedFileName(NSString *fileName) 36 | { 37 | UIDevice *device = [UIDevice currentDevice]; 38 | UIWindow *keyWindow = [[UIApplication sharedApplication] fb_strictKeyWindow]; 39 | CGSize screenSize = keyWindow.bounds.size; 40 | NSString *os = device.systemVersion; 41 | 42 | fileName = [NSString stringWithFormat:@"%@_%@%@_%.0fx%.0f", fileName, device.model, os, screenSize.width, screenSize.height]; 43 | 44 | NSMutableCharacterSet *invalidCharacters = [NSMutableCharacterSet new]; 45 | [invalidCharacters formUnionWithCharacterSet:[NSCharacterSet whitespaceCharacterSet]]; 46 | [invalidCharacters formUnionWithCharacterSet:[NSCharacterSet punctuationCharacterSet]]; 47 | NSArray *validComponents = [fileName componentsSeparatedByCharactersInSet:invalidCharacters]; 48 | fileName = [validComponents componentsJoinedByString:@"_"]; 49 | 50 | return fileName; 51 | } -------------------------------------------------------------------------------- /Example/Pods/FBSnapshotTestCase/FBSnapshotTestCase/FBSnapshotTestController.h: -------------------------------------------------------------------------------- 1 | /* 2 | * Copyright (c) 2015, Facebook, Inc. 3 | * All rights reserved. 4 | * 5 | * This source code is licensed under the BSD-style license found in the 6 | * LICENSE file in the root directory of this source tree. An additional grant 7 | * of patent rights can be found in the PATENTS file in the same directory. 8 | * 9 | */ 10 | 11 | #import 12 | #import 13 | 14 | typedef NS_ENUM(NSInteger, FBSnapshotTestControllerErrorCode) { 15 | FBSnapshotTestControllerErrorCodeUnknown, 16 | FBSnapshotTestControllerErrorCodeNeedsRecord, 17 | FBSnapshotTestControllerErrorCodePNGCreationFailed, 18 | FBSnapshotTestControllerErrorCodeImagesDifferentSizes, 19 | FBSnapshotTestControllerErrorCodeImagesDifferent, 20 | }; 21 | /** 22 | Errors returned by the methods of FBSnapshotTestController use this domain. 23 | */ 24 | extern NSString *const FBSnapshotTestControllerErrorDomain; 25 | 26 | /** 27 | Errors returned by the methods of FBSnapshotTestController sometimes contain this key in the `userInfo` dictionary. 28 | */ 29 | extern NSString *const FBReferenceImageFilePathKey; 30 | 31 | /** 32 | Errors returned by the methods of FBSnapshotTestController sometimes contain this key in the `userInfo` dictionary. 33 | */ 34 | extern NSString *const FBReferenceImageKey; 35 | 36 | /** 37 | Errors returned by the methods of FBSnapshotTestController sometimes contain this key in the `userInfo` dictionary. 38 | */ 39 | extern NSString *const FBCapturedImageKey; 40 | 41 | /** 42 | Errors returned by the methods of FBSnapshotTestController sometimes contain this key in the `userInfo` dictionary. 43 | */ 44 | extern NSString *const FBDiffedImageKey; 45 | 46 | /** 47 | Provides the heavy-lifting for FBSnapshotTestCase. It loads and saves images, along with performing the actual pixel- 48 | by-pixel comparison of images. 49 | Instances are initialized with the test class, and directories to read and write to. 50 | */ 51 | @interface FBSnapshotTestController : NSObject 52 | 53 | /** 54 | Record snapshots. 55 | */ 56 | @property (readwrite, nonatomic, assign) BOOL recordMode; 57 | 58 | /** 59 | When @c YES appends the name of the device model and OS to the snapshot file name. 60 | The default value is @c NO. 61 | */ 62 | @property (readwrite, nonatomic, assign, getter=isDeviceAgnostic) BOOL deviceAgnostic; 63 | 64 | /** 65 | Uses drawViewHierarchyInRect:afterScreenUpdates: to draw the image instead of renderInContext: 66 | */ 67 | @property (readwrite, nonatomic, assign) BOOL usesDrawViewHierarchyInRect; 68 | 69 | /** 70 | The directory in which referfence images are stored. 71 | */ 72 | @property (readwrite, nonatomic, copy) NSString *referenceImagesDirectory; 73 | 74 | /** 75 | @param testClass The subclass of FBSnapshotTestCase that is using this controller. 76 | @returns An instance of FBSnapshotTestController. 77 | */ 78 | - (instancetype)initWithTestClass:(Class)testClass; 79 | 80 | /** 81 | Designated initializer. 82 | @param testName The name of the tests. 83 | @returns An instance of FBSnapshotTestController. 84 | */ 85 | - (instancetype)initWithTestName:(NSString *)testName; 86 | 87 | /** 88 | Performs the comparison of the layer. 89 | @param layer The Layer to snapshot. 90 | @param selector The test method being run. 91 | @param identifier An optional identifier, used is there are muliptle snapshot tests in a given -test method. 92 | @param error An error to log in an XCTAssert() macro if the method fails (missing reference image, images differ, etc). 93 | @returns YES if the comparison (or saving of the reference image) succeeded. 94 | */ 95 | - (BOOL)compareSnapshotOfLayer:(CALayer *)layer 96 | selector:(SEL)selector 97 | identifier:(NSString *)identifier 98 | error:(NSError **)errorPtr; 99 | 100 | /** 101 | Performs the comparison of the view. 102 | @param view The view to snapshot. 103 | @param selector The test method being run. 104 | @param identifier An optional identifier, used is there are muliptle snapshot tests in a given -test method. 105 | @param error An error to log in an XCTAssert() macro if the method fails (missing reference image, images differ, etc). 106 | @returns YES if the comparison (or saving of the reference image) succeeded. 107 | */ 108 | - (BOOL)compareSnapshotOfView:(UIView *)view 109 | selector:(SEL)selector 110 | identifier:(NSString *)identifier 111 | error:(NSError **)errorPtr; 112 | 113 | /** 114 | Performs the comparison of a view or layer. 115 | @param view The view or layer to snapshot. 116 | @param selector The test method being run. 117 | @param identifier An optional identifier, used is there are muliptle snapshot tests in a given -test method. 118 | @param tolerance The percentage of pixels that can differ and still be considered 'identical' 119 | @param error An error to log in an XCTAssert() macro if the method fails (missing reference image, images differ, etc). 120 | @returns YES if the comparison (or saving of the reference image) succeeded. 121 | */ 122 | - (BOOL)compareSnapshotOfViewOrLayer:(id)viewOrLayer 123 | selector:(SEL)selector 124 | identifier:(NSString *)identifier 125 | tolerance:(CGFloat)tolerance 126 | error:(NSError **)errorPtr; 127 | 128 | /** 129 | Loads a reference image. 130 | @param selector The test method being run. 131 | @param identifier The optional identifier, used when multiple images are tested in a single -test method. 132 | @param errorPtr An error, if this methods returns nil, the error will be something useful. 133 | @returns An image. 134 | */ 135 | - (UIImage *)referenceImageForSelector:(SEL)selector 136 | identifier:(NSString *)identifier 137 | error:(NSError **)errorPtr; 138 | 139 | /** 140 | Performs a pixel-by-pixel comparison of the two images with an allowable margin of error. 141 | @param referenceImage The reference (correct) image. 142 | @param image The image to test against the reference. 143 | @param tolerance The percentage of pixels that can differ and still be considered 'identical' 144 | @param errorPtr An error that indicates why the comparison failed if it does. 145 | @returns YES if the comparison succeeded and the images are the same(ish). 146 | */ 147 | - (BOOL)compareReferenceImage:(UIImage *)referenceImage 148 | toImage:(UIImage *)image 149 | tolerance:(CGFloat)tolerance 150 | error:(NSError **)errorPtr; 151 | 152 | /** 153 | Saves the reference image and the test image to `failedOutputDirectory`. 154 | @param referenceImage The reference (correct) image. 155 | @param testImage The image to test against the reference. 156 | @param selector The test method being run. 157 | @param identifier The optional identifier, used when multiple images are tested in a single -test method. 158 | @param errorPtr An error that indicates why the comparison failed if it does. 159 | @returns YES if the save succeeded. 160 | */ 161 | - (BOOL)saveFailedReferenceImage:(UIImage *)referenceImage 162 | testImage:(UIImage *)testImage 163 | selector:(SEL)selector 164 | identifier:(NSString *)identifier 165 | error:(NSError **)errorPtr; 166 | @end 167 | -------------------------------------------------------------------------------- /Example/Pods/FBSnapshotTestCase/FBSnapshotTestCase/FBSnapshotTestController.m: -------------------------------------------------------------------------------- 1 | /* 2 | * Copyright (c) 2015, Facebook, Inc. 3 | * All rights reserved. 4 | * 5 | * This source code is licensed under the BSD-style license found in the 6 | * LICENSE file in the root directory of this source tree. An additional grant 7 | * of patent rights can be found in the PATENTS file in the same directory. 8 | * 9 | */ 10 | 11 | #import 12 | #import 13 | #import 14 | #import 15 | #import 16 | 17 | #import 18 | 19 | NSString *const FBSnapshotTestControllerErrorDomain = @"FBSnapshotTestControllerErrorDomain"; 20 | NSString *const FBReferenceImageFilePathKey = @"FBReferenceImageFilePathKey"; 21 | NSString *const FBReferenceImageKey = @"FBReferenceImageKey"; 22 | NSString *const FBCapturedImageKey = @"FBCapturedImageKey"; 23 | NSString *const FBDiffedImageKey = @"FBDiffedImageKey"; 24 | 25 | typedef NS_ENUM(NSUInteger, FBTestSnapshotFileNameType) { 26 | FBTestSnapshotFileNameTypeReference, 27 | FBTestSnapshotFileNameTypeFailedReference, 28 | FBTestSnapshotFileNameTypeFailedTest, 29 | FBTestSnapshotFileNameTypeFailedTestDiff, 30 | }; 31 | 32 | @implementation FBSnapshotTestController 33 | { 34 | NSString *_testName; 35 | NSFileManager *_fileManager; 36 | } 37 | 38 | #pragma mark - Initializers 39 | 40 | - (instancetype)initWithTestClass:(Class)testClass; 41 | { 42 | return [self initWithTestName:NSStringFromClass(testClass)]; 43 | } 44 | 45 | - (instancetype)initWithTestName:(NSString *)testName 46 | { 47 | if (self = [super init]) { 48 | _testName = [testName copy]; 49 | _deviceAgnostic = NO; 50 | 51 | _fileManager = [[NSFileManager alloc] init]; 52 | } 53 | return self; 54 | } 55 | 56 | #pragma mark - Overrides 57 | 58 | - (NSString *)description 59 | { 60 | return [NSString stringWithFormat:@"%@ %@", [super description], _referenceImagesDirectory]; 61 | } 62 | 63 | #pragma mark - Public API 64 | 65 | - (BOOL)compareSnapshotOfLayer:(CALayer *)layer 66 | selector:(SEL)selector 67 | identifier:(NSString *)identifier 68 | error:(NSError **)errorPtr 69 | { 70 | return [self compareSnapshotOfViewOrLayer:layer 71 | selector:selector 72 | identifier:identifier 73 | tolerance:0 74 | error:errorPtr]; 75 | } 76 | 77 | - (BOOL)compareSnapshotOfView:(UIView *)view 78 | selector:(SEL)selector 79 | identifier:(NSString *)identifier 80 | error:(NSError **)errorPtr 81 | { 82 | return [self compareSnapshotOfViewOrLayer:view 83 | selector:selector 84 | identifier:identifier 85 | tolerance:0 86 | error:errorPtr]; 87 | } 88 | 89 | - (BOOL)compareSnapshotOfViewOrLayer:(id)viewOrLayer 90 | selector:(SEL)selector 91 | identifier:(NSString *)identifier 92 | tolerance:(CGFloat)tolerance 93 | error:(NSError **)errorPtr 94 | { 95 | if (self.recordMode) { 96 | return [self _recordSnapshotOfViewOrLayer:viewOrLayer selector:selector identifier:identifier error:errorPtr]; 97 | } else { 98 | return [self _performPixelComparisonWithViewOrLayer:viewOrLayer selector:selector identifier:identifier tolerance:tolerance error:errorPtr]; 99 | } 100 | } 101 | 102 | - (UIImage *)referenceImageForSelector:(SEL)selector 103 | identifier:(NSString *)identifier 104 | error:(NSError **)errorPtr 105 | { 106 | NSString *filePath = [self _referenceFilePathForSelector:selector identifier:identifier]; 107 | UIImage *image = [UIImage imageWithContentsOfFile:filePath]; 108 | if (nil == image && NULL != errorPtr) { 109 | BOOL exists = [_fileManager fileExistsAtPath:filePath]; 110 | if (!exists) { 111 | *errorPtr = [NSError errorWithDomain:FBSnapshotTestControllerErrorDomain 112 | code:FBSnapshotTestControllerErrorCodeNeedsRecord 113 | userInfo:@{ 114 | FBReferenceImageFilePathKey: filePath, 115 | NSLocalizedDescriptionKey: @"Unable to load reference image.", 116 | NSLocalizedFailureReasonErrorKey: @"Reference image not found. You need to run the test in record mode", 117 | }]; 118 | } else { 119 | *errorPtr = [NSError errorWithDomain:FBSnapshotTestControllerErrorDomain 120 | code:FBSnapshotTestControllerErrorCodeUnknown 121 | userInfo:nil]; 122 | } 123 | } 124 | return image; 125 | } 126 | 127 | - (BOOL)compareReferenceImage:(UIImage *)referenceImage 128 | toImage:(UIImage *)image 129 | tolerance:(CGFloat)tolerance 130 | error:(NSError **)errorPtr 131 | { 132 | BOOL sameImageDimensions = CGSizeEqualToSize(referenceImage.size, image.size); 133 | if (sameImageDimensions && [referenceImage fb_compareWithImage:image tolerance:tolerance]) { 134 | return YES; 135 | } 136 | 137 | if (NULL != errorPtr) { 138 | NSString *errorDescription = sameImageDimensions ? @"Images different" : @"Images different sizes"; 139 | NSString *errorReason = sameImageDimensions ? [NSString stringWithFormat:@"image pixels differed by more than %.2f%% from the reference image", tolerance * 100] 140 | : [NSString stringWithFormat:@"referenceImage:%@, image:%@", NSStringFromCGSize(referenceImage.size), NSStringFromCGSize(image.size)]; 141 | FBSnapshotTestControllerErrorCode errorCode = sameImageDimensions ? FBSnapshotTestControllerErrorCodeImagesDifferent : FBSnapshotTestControllerErrorCodeImagesDifferentSizes; 142 | 143 | *errorPtr = [NSError errorWithDomain:FBSnapshotTestControllerErrorDomain 144 | code:errorCode 145 | userInfo:@{ 146 | NSLocalizedDescriptionKey: errorDescription, 147 | NSLocalizedFailureReasonErrorKey: errorReason, 148 | FBReferenceImageKey: referenceImage, 149 | FBCapturedImageKey: image, 150 | FBDiffedImageKey: [referenceImage fb_diffWithImage:image], 151 | }]; 152 | } 153 | return NO; 154 | } 155 | 156 | - (BOOL)saveFailedReferenceImage:(UIImage *)referenceImage 157 | testImage:(UIImage *)testImage 158 | selector:(SEL)selector 159 | identifier:(NSString *)identifier 160 | error:(NSError **)errorPtr 161 | { 162 | NSData *referencePNGData = UIImagePNGRepresentation(referenceImage); 163 | NSData *testPNGData = UIImagePNGRepresentation(testImage); 164 | 165 | NSString *referencePath = [self _failedFilePathForSelector:selector 166 | identifier:identifier 167 | fileNameType:FBTestSnapshotFileNameTypeFailedReference]; 168 | 169 | NSError *creationError = nil; 170 | BOOL didCreateDir = [_fileManager createDirectoryAtPath:[referencePath stringByDeletingLastPathComponent] 171 | withIntermediateDirectories:YES 172 | attributes:nil 173 | error:&creationError]; 174 | if (!didCreateDir) { 175 | if (NULL != errorPtr) { 176 | *errorPtr = creationError; 177 | } 178 | return NO; 179 | } 180 | 181 | if (![referencePNGData writeToFile:referencePath options:NSDataWritingAtomic error:errorPtr]) { 182 | return NO; 183 | } 184 | 185 | NSString *testPath = [self _failedFilePathForSelector:selector 186 | identifier:identifier 187 | fileNameType:FBTestSnapshotFileNameTypeFailedTest]; 188 | 189 | if (![testPNGData writeToFile:testPath options:NSDataWritingAtomic error:errorPtr]) { 190 | return NO; 191 | } 192 | 193 | NSString *diffPath = [self _failedFilePathForSelector:selector 194 | identifier:identifier 195 | fileNameType:FBTestSnapshotFileNameTypeFailedTestDiff]; 196 | 197 | UIImage *diffImage = [referenceImage fb_diffWithImage:testImage]; 198 | NSData *diffImageData = UIImagePNGRepresentation(diffImage); 199 | 200 | if (![diffImageData writeToFile:diffPath options:NSDataWritingAtomic error:errorPtr]) { 201 | return NO; 202 | } 203 | 204 | NSLog(@"If you have Kaleidoscope installed you can run this command to see an image diff:\n" 205 | @"ksdiff \"%@\" \"%@\"", referencePath, testPath); 206 | 207 | return YES; 208 | } 209 | 210 | #pragma mark - Private API 211 | 212 | - (NSString *)_fileNameForSelector:(SEL)selector 213 | identifier:(NSString *)identifier 214 | fileNameType:(FBTestSnapshotFileNameType)fileNameType 215 | { 216 | NSString *fileName = nil; 217 | switch (fileNameType) { 218 | case FBTestSnapshotFileNameTypeFailedReference: 219 | fileName = @"reference_"; 220 | break; 221 | case FBTestSnapshotFileNameTypeFailedTest: 222 | fileName = @"failed_"; 223 | break; 224 | case FBTestSnapshotFileNameTypeFailedTestDiff: 225 | fileName = @"diff_"; 226 | break; 227 | default: 228 | fileName = @""; 229 | break; 230 | } 231 | fileName = [fileName stringByAppendingString:NSStringFromSelector(selector)]; 232 | if (0 < identifier.length) { 233 | fileName = [fileName stringByAppendingFormat:@"_%@", identifier]; 234 | } 235 | 236 | if (self.isDeviceAgnostic) { 237 | fileName = FBDeviceAgnosticNormalizedFileName(fileName); 238 | } 239 | 240 | if ([[UIScreen mainScreen] scale] > 1) { 241 | fileName = [fileName stringByAppendingFormat:@"@%.fx", [[UIScreen mainScreen] scale]]; 242 | } 243 | fileName = [fileName stringByAppendingPathExtension:@"png"]; 244 | return fileName; 245 | } 246 | 247 | - (NSString *)_referenceFilePathForSelector:(SEL)selector 248 | identifier:(NSString *)identifier 249 | { 250 | NSString *fileName = [self _fileNameForSelector:selector 251 | identifier:identifier 252 | fileNameType:FBTestSnapshotFileNameTypeReference]; 253 | NSString *filePath = [_referenceImagesDirectory stringByAppendingPathComponent:_testName]; 254 | filePath = [filePath stringByAppendingPathComponent:fileName]; 255 | return filePath; 256 | } 257 | 258 | - (NSString *)_failedFilePathForSelector:(SEL)selector 259 | identifier:(NSString *)identifier 260 | fileNameType:(FBTestSnapshotFileNameType)fileNameType 261 | { 262 | NSString *fileName = [self _fileNameForSelector:selector 263 | identifier:identifier 264 | fileNameType:fileNameType]; 265 | NSString *folderPath = NSTemporaryDirectory(); 266 | if (getenv("IMAGE_DIFF_DIR")) { 267 | folderPath = @(getenv("IMAGE_DIFF_DIR")); 268 | } 269 | NSString *filePath = [folderPath stringByAppendingPathComponent:_testName]; 270 | filePath = [filePath stringByAppendingPathComponent:fileName]; 271 | return filePath; 272 | } 273 | 274 | - (BOOL)_performPixelComparisonWithViewOrLayer:(id)viewOrLayer 275 | selector:(SEL)selector 276 | identifier:(NSString *)identifier 277 | tolerance:(CGFloat)tolerance 278 | error:(NSError **)errorPtr 279 | { 280 | UIImage *referenceImage = [self referenceImageForSelector:selector identifier:identifier error:errorPtr]; 281 | if (nil != referenceImage) { 282 | UIImage *snapshot = [self _imageForViewOrLayer:viewOrLayer]; 283 | BOOL imagesSame = [self compareReferenceImage:referenceImage toImage:snapshot tolerance:tolerance error:errorPtr]; 284 | if (!imagesSame) { 285 | NSError *saveError = nil; 286 | if ([self saveFailedReferenceImage:referenceImage testImage:snapshot selector:selector identifier:identifier error:&saveError] == NO) { 287 | NSLog(@"Error saving test images: %@", saveError); 288 | } 289 | } 290 | return imagesSame; 291 | } 292 | return NO; 293 | } 294 | 295 | - (BOOL)_recordSnapshotOfViewOrLayer:(id)viewOrLayer 296 | selector:(SEL)selector 297 | identifier:(NSString *)identifier 298 | error:(NSError **)errorPtr 299 | { 300 | UIImage *snapshot = [self _imageForViewOrLayer:viewOrLayer]; 301 | return [self _saveReferenceImage:snapshot selector:selector identifier:identifier error:errorPtr]; 302 | } 303 | 304 | - (BOOL)_saveReferenceImage:(UIImage *)image 305 | selector:(SEL)selector 306 | identifier:(NSString *)identifier 307 | error:(NSError **)errorPtr 308 | { 309 | BOOL didWrite = NO; 310 | if (nil != image) { 311 | NSString *filePath = [self _referenceFilePathForSelector:selector identifier:identifier]; 312 | NSData *pngData = UIImagePNGRepresentation(image); 313 | if (nil != pngData) { 314 | NSError *creationError = nil; 315 | BOOL didCreateDir = [_fileManager createDirectoryAtPath:[filePath stringByDeletingLastPathComponent] 316 | withIntermediateDirectories:YES 317 | attributes:nil 318 | error:&creationError]; 319 | if (!didCreateDir) { 320 | if (NULL != errorPtr) { 321 | *errorPtr = creationError; 322 | } 323 | return NO; 324 | } 325 | didWrite = [pngData writeToFile:filePath options:NSDataWritingAtomic error:errorPtr]; 326 | if (didWrite) { 327 | NSLog(@"Reference image save at: %@", filePath); 328 | } 329 | } else { 330 | if (nil != errorPtr) { 331 | *errorPtr = [NSError errorWithDomain:FBSnapshotTestControllerErrorDomain 332 | code:FBSnapshotTestControllerErrorCodePNGCreationFailed 333 | userInfo:@{ 334 | FBReferenceImageFilePathKey: filePath, 335 | }]; 336 | } 337 | } 338 | } 339 | return didWrite; 340 | } 341 | 342 | - (UIImage *)_imageForViewOrLayer:(id)viewOrLayer 343 | { 344 | if ([viewOrLayer isKindOfClass:[UIView class]]) { 345 | if (_usesDrawViewHierarchyInRect) { 346 | return [UIImage fb_imageForView:viewOrLayer]; 347 | } else { 348 | return [UIImage fb_imageForViewLayer:viewOrLayer]; 349 | } 350 | } else if ([viewOrLayer isKindOfClass:[CALayer class]]) { 351 | return [UIImage fb_imageForLayer:viewOrLayer]; 352 | } else { 353 | [NSException raise:@"Only UIView and CALayer classes can be snapshotted" format:@"%@", viewOrLayer]; 354 | } 355 | return nil; 356 | } 357 | 358 | @end 359 | -------------------------------------------------------------------------------- /Example/Pods/FBSnapshotTestCase/FBSnapshotTestCase/SwiftSupport.swift: -------------------------------------------------------------------------------- 1 | /* 2 | * Copyright (c) 2015, Facebook, Inc. 3 | * All rights reserved. 4 | * 5 | * This source code is licensed under the BSD-style license found in the 6 | * LICENSE file in the root directory of this source tree. An additional grant 7 | * of patent rights can be found in the PATENTS file in the same directory. 8 | * 9 | */ 10 | 11 | #if swift(>=3) 12 | public extension FBSnapshotTestCase { 13 | public func FBSnapshotVerifyView(_ view: UIView, identifier: String = "", suffixes: NSOrderedSet = FBSnapshotTestCaseDefaultSuffixes(), tolerance: CGFloat = 0, file: StaticString = #file, line: UInt = #line) { 14 | FBSnapshotVerifyViewOrLayer(view, identifier: identifier, suffixes: suffixes, tolerance: tolerance, file: file, line: line) 15 | } 16 | 17 | public func FBSnapshotVerifyLayer(_ layer: CALayer, identifier: String = "", suffixes: NSOrderedSet = FBSnapshotTestCaseDefaultSuffixes(), tolerance: CGFloat = 0, file: StaticString = #file, line: UInt = #line) { 18 | FBSnapshotVerifyViewOrLayer(layer, identifier: identifier, suffixes: suffixes, tolerance: tolerance, file: file, line: line) 19 | } 20 | 21 | private func FBSnapshotVerifyViewOrLayer(_ viewOrLayer: AnyObject, identifier: String = "", suffixes: NSOrderedSet = FBSnapshotTestCaseDefaultSuffixes(), tolerance: CGFloat = 0, file: StaticString = #file, line: UInt = #line) { 22 | let envReferenceImageDirectory = self.getReferenceImageDirectory(withDefault: FB_REFERENCE_IMAGE_DIR) 23 | var error: NSError? 24 | var comparisonSuccess = false 25 | 26 | if let envReferenceImageDirectory = envReferenceImageDirectory { 27 | for suffix in suffixes { 28 | let referenceImagesDirectory = "\(envReferenceImageDirectory)\(suffix)" 29 | if viewOrLayer.isKind(of: UIView.self) { 30 | do { 31 | try compareSnapshot(of: viewOrLayer as! UIView, referenceImagesDirectory: referenceImagesDirectory, identifier: identifier, tolerance: tolerance) 32 | comparisonSuccess = true 33 | } catch let error1 as NSError { 34 | error = error1 35 | comparisonSuccess = false 36 | } 37 | } else if viewOrLayer.isKind(of: CALayer.self) { 38 | do { 39 | try compareSnapshot(of: viewOrLayer as! CALayer, referenceImagesDirectory: referenceImagesDirectory, identifier: identifier, tolerance: tolerance) 40 | comparisonSuccess = true 41 | } catch let error1 as NSError { 42 | error = error1 43 | comparisonSuccess = false 44 | } 45 | } else { 46 | assertionFailure("Only UIView and CALayer classes can be snapshotted") 47 | } 48 | 49 | assert(recordMode == false, message: "Test ran in record mode. Reference image is now saved. Disable record mode to perform an actual snapshot comparison!", file: file, line: line) 50 | 51 | if comparisonSuccess || recordMode { 52 | break 53 | } 54 | 55 | assert(comparisonSuccess, message: "Snapshot comparison failed: \(error)", file: file, line: line) 56 | } 57 | } else { 58 | XCTFail("Missing value for referenceImagesDirectory - Set FB_REFERENCE_IMAGE_DIR as Environment variable in your scheme.") 59 | } 60 | } 61 | 62 | func assert(_ assertion: Bool, message: String, file: StaticString, line: UInt) { 63 | if !assertion { 64 | XCTFail(message, file: file, line: line) 65 | } 66 | } 67 | } 68 | #else 69 | public extension FBSnapshotTestCase { 70 | public func FBSnapshotVerifyView(view: UIView, identifier: String = "", suffixes: NSOrderedSet = FBSnapshotTestCaseDefaultSuffixes(), tolerance: CGFloat = 0, file: StaticString = #file, line: UInt = #line) { 71 | FBSnapshotVerifyViewOrLayer(view, identifier: identifier, suffixes: suffixes, tolerance: tolerance, file: file, line: line) 72 | } 73 | 74 | public func FBSnapshotVerifyLayer(layer: CALayer, identifier: String = "", suffixes: NSOrderedSet = FBSnapshotTestCaseDefaultSuffixes(), tolerance: CGFloat = 0, file: StaticString = #file, line: UInt = #line) { 75 | FBSnapshotVerifyViewOrLayer(layer, identifier: identifier, suffixes: suffixes, tolerance: tolerance, file: file, line: line) 76 | } 77 | 78 | private func FBSnapshotVerifyViewOrLayer(viewOrLayer: AnyObject, identifier: String = "", suffixes: NSOrderedSet = FBSnapshotTestCaseDefaultSuffixes(), tolerance: CGFloat = 0, file: StaticString = #file, line: UInt = #line) { 79 | let envReferenceImageDirectory = self.getReferenceImageDirectoryWithDefault(FB_REFERENCE_IMAGE_DIR) 80 | var error: NSError? 81 | var comparisonSuccess = false 82 | 83 | if let envReferenceImageDirectory = envReferenceImageDirectory { 84 | for suffix in suffixes { 85 | let referenceImagesDirectory = "\(envReferenceImageDirectory)\(suffix)" 86 | if viewOrLayer.isKindOfClass(UIView) { 87 | do { 88 | try compareSnapshotOfView(viewOrLayer as! UIView, referenceImagesDirectory: referenceImagesDirectory, identifier: identifier, tolerance: tolerance) 89 | comparisonSuccess = true 90 | } catch let error1 as NSError { 91 | error = error1 92 | comparisonSuccess = false 93 | } 94 | } else if viewOrLayer.isKindOfClass(CALayer) { 95 | do { 96 | try compareSnapshotOfLayer(viewOrLayer as! CALayer, referenceImagesDirectory: referenceImagesDirectory, identifier: identifier, tolerance: tolerance) 97 | comparisonSuccess = true 98 | } catch let error1 as NSError { 99 | error = error1 100 | comparisonSuccess = false 101 | } 102 | } else { 103 | assertionFailure("Only UIView and CALayer classes can be snapshotted") 104 | } 105 | 106 | assert(recordMode == false, message: "Test ran in record mode. Reference image is now saved. Disable record mode to perform an actual snapshot comparison!", file: file, line: line) 107 | 108 | if comparisonSuccess || recordMode { 109 | break 110 | } 111 | 112 | assert(comparisonSuccess, message: "Snapshot comparison failed: \(error)", file: file, line: line) 113 | } 114 | } else { 115 | XCTFail("Missing value for referenceImagesDirectory - Set FB_REFERENCE_IMAGE_DIR as Environment variable in your scheme.") 116 | } 117 | } 118 | 119 | func assert(assertion: Bool, message: String, file: StaticString, line: UInt) { 120 | if !assertion { 121 | XCTFail(message, file: file, line: line) 122 | } 123 | } 124 | } 125 | #endif 126 | -------------------------------------------------------------------------------- /Example/Pods/FBSnapshotTestCase/LICENSE: -------------------------------------------------------------------------------- 1 | BSD License 2 | 3 | For the FBSnapshotTestCase software 4 | 5 | Copyright (c) 2013, Facebook, Inc. 6 | All rights reserved. 7 | 8 | Redistribution and use in source and binary forms, with or without 9 | modification, are permitted provided that the following conditions are met: 10 | 11 | * Redistributions of source code must retain the above copyright notice, 12 | this list of conditions and the following disclaimer. 13 | * Redistributions in binary form must reproduce the above copyright notice, 14 | this list of conditions and the following disclaimer in the documentation 15 | and/or other materials provided with the distribution. 16 | * Neither the name Facebook nor the names of its contributors may be used to 17 | endorse or promote products derived from this software without specific 18 | prior written permission. 19 | 20 | THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" 21 | AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 22 | IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE 23 | DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE 24 | FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 25 | DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR 26 | SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER 27 | CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, 28 | OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE 29 | OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 30 | -------------------------------------------------------------------------------- /Example/Pods/FBSnapshotTestCase/README.md: -------------------------------------------------------------------------------- 1 | FBSnapshotTestCase 2 | ====================== 3 | 4 | [![Build Status](https://travis-ci.org/facebook/ios-snapshot-test-case.svg)](https://travis-ci.org/facebook/ios-snapshot-test-case) [![Cocoa Pod Version](https://cocoapod-badges.herokuapp.com/v/FBSnapshotTestCase/badge.svg)](http://cocoadocs.org/docsets/FBSnapshotTestCase/) 5 | 6 | What it does 7 | ------------ 8 | 9 | A "snapshot test case" takes a configured `UIView` or `CALayer` and uses the 10 | `renderInContext:` method to get an image snapshot of its contents. It 11 | compares this snapshot to a "reference image" stored in your source code 12 | repository and fails the test if the two images don't match. 13 | 14 | Why? 15 | ---- 16 | 17 | At Facebook we write a lot of UI code. As you might imagine, each type of 18 | feed story is rendered using a subclass of `UIView`. There are a lot of edge 19 | cases that we want to handle correctly: 20 | 21 | - What if there is more text than can fit in the space available? 22 | - What if an image doesn't match the size of an image view? 23 | - What should the highlighted state look like? 24 | 25 | It's straightforward to test logic code, but less obvious how you should test 26 | views. You can do a lot of rectangle asserts, but these are hard to understand 27 | or visualize. Looking at an image diff shows you exactly what changed and how 28 | it will look to users. 29 | 30 | We developed `FBSnapshotTestCase` to make snapshot tests easy. 31 | 32 | Installation with CocoaPods 33 | --------------------------- 34 | 35 | 1. Add the following lines to your Podfile: 36 | 37 | ``` 38 | target "Tests" do 39 | pod 'FBSnapshotTestCase' 40 | end 41 | ``` 42 | 43 | If you support iOS 7 use `FBSnapshotTestCase/Core` instead, which doesn't contain Swift support. 44 | 45 | Replace "Tests" with the name of your test project. 46 | 47 | 2. There are [three ways](https://github.com/facebook/ios-snapshot-test-case/blob/master/FBSnapshotTestCase/FBSnapshotTestCase.h#L19-L29) of setting reference image directories, the recommended one is to define `FB_REFERENCE_IMAGE_DIR` in your scheme. This should point to the directory where you want reference images to be stored. At Facebook, we normally use this: 48 | 49 | |Name|Value| 50 | |:---|:----| 51 | |`FB_REFERENCE_IMAGE_DIR`|`$(SOURCE_ROOT)/$(PROJECT_NAME)Tests/ReferenceImages`| 52 | 53 | 54 | ![](FBSnapshotTestCaseDemo/Scheme_FB_REFERENCE_IMAGE_DIR.png) 55 | 56 | Creating a snapshot test 57 | ------------------------ 58 | 59 | 1. Subclass `FBSnapshotTestCase` instead of `XCTestCase`. 60 | 2. From within your test, use `FBSnapshotVerifyView`. 61 | 3. Run the test once with `self.recordMode = YES;` in the test's `-setUp` 62 | method. (This creates the reference images on disk.) 63 | 4. Remove the line enabling record mode and run the test. 64 | 65 | Features 66 | -------- 67 | 68 | - Automatically names reference images on disk according to test class and 69 | selector. 70 | - Prints a descriptive error message to the console on failure. (Bonus: 71 | failure message includes a one-line command to see an image diff if 72 | you have [Kaleidoscope](http://www.kaleidoscopeapp.com) installed.) 73 | - Supply an optional "identifier" if you want to perform multiple snapshots 74 | in a single test method. 75 | - Support for `CALayer` via `FBSnapshotVerifyLayer`. 76 | - `usesDrawViewHierarchyInRect` to handle cases like `UIVisualEffect`, `UIAppearance` and Size Classes. 77 | - `isDeviceAgnostic` to allow appending the device model (`iPhone`, `iPad`, `iPod Touch`, etc), OS version and screen size to the images (allowing to have multiple tests for the same «snapshot» for different `OS`s and devices). 78 | 79 | Notes 80 | ----- 81 | 82 | Your unit test must be an "application test", not a "logic test." (That is, it 83 | must be run within the Simulator so that it has access to UIKit.) In Xcode 5 84 | and later new projects only offer application tests, but older projects will 85 | have separate targets for the two types. 86 | 87 | Authors 88 | ------- 89 | 90 | `FBSnapshotTestCase` was written at Facebook by 91 | [Jonathan Dann](https://facebook.com/j.p.dann) with significant contributions by 92 | [Todd Krabach](https://facebook.com/toddkrabach). 93 | 94 | License 95 | ------- 96 | 97 | `FBSnapshotTestCase` is BSD-licensed. See `LICENSE`. 98 | -------------------------------------------------------------------------------- /Example/Pods/Local Podspecs/ZLBridge.podspec.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "ZLBridge", 3 | "version": "1.0.0", 4 | "summary": "A short description of ZLBridge.", 5 | "description": "TODO: Add long description of the pod here.", 6 | "homepage": "https://github.com/FPJack/ZLBridge.git", 7 | "license": { 8 | "type": "MIT", 9 | "file": "LICENSE" 10 | }, 11 | "authors": { 12 | "范鹏": "2551412939@qq.com" 13 | }, 14 | "source": { 15 | "git": "https://github.com/FPJack/ZLBridge.git", 16 | "tag": "1.0.0" 17 | }, 18 | "platforms": { 19 | "ios": "9.0" 20 | }, 21 | "source_files": "ZLBridge/Classes/**/*", 22 | "resource_bundles": { 23 | "ZLBridge": [ 24 | "ZLBridge/Assets/*" 25 | ] 26 | } 27 | } 28 | -------------------------------------------------------------------------------- /Example/Pods/Manifest.lock: -------------------------------------------------------------------------------- 1 | PODS: 2 | - FBSnapshotTestCase (2.1.4): 3 | - FBSnapshotTestCase/SwiftSupport (= 2.1.4) 4 | - FBSnapshotTestCase/Core (2.1.4) 5 | - FBSnapshotTestCase/SwiftSupport (2.1.4): 6 | - FBSnapshotTestCase/Core 7 | - ZLBridge (1.0.0) 8 | 9 | DEPENDENCIES: 10 | - FBSnapshotTestCase 11 | - ZLBridge (from `../`) 12 | 13 | SPEC REPOS: 14 | trunk: 15 | - FBSnapshotTestCase 16 | 17 | EXTERNAL SOURCES: 18 | ZLBridge: 19 | :path: "../" 20 | 21 | SPEC CHECKSUMS: 22 | FBSnapshotTestCase: 094f9f314decbabe373b87cc339bea235a63e07a 23 | ZLBridge: 5202001bbf512ad35072d1c993f55ffff9247a6b 24 | 25 | PODFILE CHECKSUM: affccafbf7f67f785a478373a0839b65dfda2bc8 26 | 27 | COCOAPODS: 1.9.3 28 | -------------------------------------------------------------------------------- /Example/Pods/Pods.xcodeproj/xcuserdata/fanpeng.xcuserdatad/xcschemes/FBSnapshotTestCase.xcscheme: -------------------------------------------------------------------------------- 1 | 2 | 5 | 8 | 9 | 15 | 21 | 22 | 23 | 24 | 25 | 30 | 31 | 32 | 33 | 43 | 44 | 45 | 46 | 52 | 53 | 55 | 56 | 59 | 60 | 61 | -------------------------------------------------------------------------------- /Example/Pods/Pods.xcodeproj/xcuserdata/fanpeng.xcuserdatad/xcschemes/Pods-ZLBridge_Example.xcscheme: -------------------------------------------------------------------------------- 1 | 2 | 5 | 8 | 9 | 15 | 21 | 22 | 23 | 24 | 25 | 30 | 31 | 32 | 33 | 43 | 44 | 50 | 51 | 53 | 54 | 57 | 58 | 59 | -------------------------------------------------------------------------------- /Example/Pods/Pods.xcodeproj/xcuserdata/fanpeng.xcuserdatad/xcschemes/Pods-ZLBridge_Tests.xcscheme: -------------------------------------------------------------------------------- 1 | 2 | 5 | 8 | 9 | 15 | 21 | 22 | 23 | 24 | 25 | 30 | 31 | 32 | 33 | 43 | 44 | 50 | 51 | 53 | 54 | 57 | 58 | 59 | -------------------------------------------------------------------------------- /Example/Pods/Pods.xcodeproj/xcuserdata/fanpeng.xcuserdatad/xcschemes/ZLBridge-ZLBridge.xcscheme: -------------------------------------------------------------------------------- 1 | 2 | 5 | 8 | 9 | 15 | 21 | 22 | 23 | 24 | 25 | 30 | 31 | 32 | 33 | 43 | 44 | 50 | 51 | 53 | 54 | 57 | 58 | 59 | -------------------------------------------------------------------------------- /Example/Pods/Pods.xcodeproj/xcuserdata/fanpeng.xcuserdatad/xcschemes/ZLBridge.xcscheme: -------------------------------------------------------------------------------- 1 | 2 | 5 | 8 | 9 | 15 | 21 | 22 | 23 | 24 | 25 | 30 | 31 | 32 | 33 | 43 | 44 | 45 | 46 | 52 | 53 | 55 | 56 | 59 | 60 | 61 | -------------------------------------------------------------------------------- /Example/Pods/Pods.xcodeproj/xcuserdata/fanpeng.xcuserdatad/xcschemes/xcschememanagement.plist: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | SchemeUserState 6 | 7 | FBSnapshotTestCase.xcscheme 8 | 9 | isShown 10 | 11 | 12 | Pods-ZLBridge_Example.xcscheme 13 | 14 | isShown 15 | 16 | 17 | Pods-ZLBridge_Tests.xcscheme 18 | 19 | isShown 20 | 21 | 22 | ZLBridge-ZLBridge.xcscheme 23 | 24 | isShown 25 | 26 | 27 | ZLBridge.xcscheme 28 | 29 | isShown 30 | 31 | 32 | 33 | SuppressBuildableAutocreation 34 | 35 | 36 | 37 | -------------------------------------------------------------------------------- /Example/Pods/Target Support Files/FBSnapshotTestCase/FBSnapshotTestCase-Info.plist: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | CFBundleDevelopmentRegion 6 | en 7 | CFBundleExecutable 8 | ${EXECUTABLE_NAME} 9 | CFBundleIdentifier 10 | ${PRODUCT_BUNDLE_IDENTIFIER} 11 | CFBundleInfoDictionaryVersion 12 | 6.0 13 | CFBundleName 14 | ${PRODUCT_NAME} 15 | CFBundlePackageType 16 | FMWK 17 | CFBundleShortVersionString 18 | 2.1.4 19 | CFBundleSignature 20 | ???? 21 | CFBundleVersion 22 | ${CURRENT_PROJECT_VERSION} 23 | NSPrincipalClass 24 | 25 | 26 | 27 | -------------------------------------------------------------------------------- /Example/Pods/Target Support Files/FBSnapshotTestCase/FBSnapshotTestCase-dummy.m: -------------------------------------------------------------------------------- 1 | #import 2 | @interface PodsDummy_FBSnapshotTestCase : NSObject 3 | @end 4 | @implementation PodsDummy_FBSnapshotTestCase 5 | @end 6 | -------------------------------------------------------------------------------- /Example/Pods/Target Support Files/FBSnapshotTestCase/FBSnapshotTestCase-prefix.pch: -------------------------------------------------------------------------------- 1 | #ifdef __OBJC__ 2 | #import 3 | #else 4 | #ifndef FOUNDATION_EXPORT 5 | #if defined(__cplusplus) 6 | #define FOUNDATION_EXPORT extern "C" 7 | #else 8 | #define FOUNDATION_EXPORT extern 9 | #endif 10 | #endif 11 | #endif 12 | 13 | -------------------------------------------------------------------------------- /Example/Pods/Target Support Files/FBSnapshotTestCase/FBSnapshotTestCase-umbrella.h: -------------------------------------------------------------------------------- 1 | #ifdef __OBJC__ 2 | #import 3 | #else 4 | #ifndef FOUNDATION_EXPORT 5 | #if defined(__cplusplus) 6 | #define FOUNDATION_EXPORT extern "C" 7 | #else 8 | #define FOUNDATION_EXPORT extern 9 | #endif 10 | #endif 11 | #endif 12 | 13 | #import "FBSnapshotTestCase.h" 14 | #import "FBSnapshotTestCasePlatform.h" 15 | #import "FBSnapshotTestController.h" 16 | 17 | FOUNDATION_EXPORT double FBSnapshotTestCaseVersionNumber; 18 | FOUNDATION_EXPORT const unsigned char FBSnapshotTestCaseVersionString[]; 19 | 20 | -------------------------------------------------------------------------------- /Example/Pods/Target Support Files/FBSnapshotTestCase/FBSnapshotTestCase.debug.xcconfig: -------------------------------------------------------------------------------- 1 | CONFIGURATION_BUILD_DIR = ${PODS_CONFIGURATION_BUILD_DIR}/FBSnapshotTestCase 2 | ENABLE_BITCODE = NO 3 | FRAMEWORK_SEARCH_PATHS = $(inherited) "$(PLATFORM_DIR)/Developer/Library/Frameworks" 4 | GCC_PREPROCESSOR_DEFINITIONS = $(inherited) COCOAPODS=1 5 | LIBRARY_SEARCH_PATHS = $(inherited) "$(PLATFORM_DIR)/Developer/usr/lib" 6 | OTHER_LDFLAGS = $(inherited) -framework "Foundation" -framework "QuartzCore" -framework "UIKit" -framework "XCTest" 7 | OTHER_SWIFT_FLAGS = $(inherited) -D COCOAPODS 8 | PODS_BUILD_DIR = ${BUILD_DIR} 9 | PODS_CONFIGURATION_BUILD_DIR = ${PODS_BUILD_DIR}/$(CONFIGURATION)$(EFFECTIVE_PLATFORM_NAME) 10 | PODS_ROOT = ${SRCROOT} 11 | PODS_TARGET_SRCROOT = ${PODS_ROOT}/FBSnapshotTestCase 12 | PRODUCT_BUNDLE_IDENTIFIER = org.cocoapods.${PRODUCT_NAME:rfc1034identifier} 13 | SKIP_INSTALL = YES 14 | SWIFT_INCLUDE_PATHS = $(inherited) "$(PLATFORM_DIR)/Developer/usr/lib" 15 | SYSTEM_FRAMEWORK_SEARCH_PATHS = $(inherited) "$(PLATFORM_DIR)/Developer/Library/Frameworks" 16 | USE_RECURSIVE_SCRIPT_INPUTS_IN_SCRIPT_PHASES = YES 17 | -------------------------------------------------------------------------------- /Example/Pods/Target Support Files/FBSnapshotTestCase/FBSnapshotTestCase.modulemap: -------------------------------------------------------------------------------- 1 | framework module FBSnapshotTestCase { 2 | umbrella header "FBSnapshotTestCase-umbrella.h" 3 | 4 | export * 5 | module * { export * } 6 | } 7 | -------------------------------------------------------------------------------- /Example/Pods/Target Support Files/FBSnapshotTestCase/FBSnapshotTestCase.release.xcconfig: -------------------------------------------------------------------------------- 1 | CONFIGURATION_BUILD_DIR = ${PODS_CONFIGURATION_BUILD_DIR}/FBSnapshotTestCase 2 | ENABLE_BITCODE = NO 3 | FRAMEWORK_SEARCH_PATHS = $(inherited) "$(PLATFORM_DIR)/Developer/Library/Frameworks" 4 | GCC_PREPROCESSOR_DEFINITIONS = $(inherited) COCOAPODS=1 5 | LIBRARY_SEARCH_PATHS = $(inherited) "$(PLATFORM_DIR)/Developer/usr/lib" 6 | OTHER_LDFLAGS = $(inherited) -framework "Foundation" -framework "QuartzCore" -framework "UIKit" -framework "XCTest" 7 | OTHER_SWIFT_FLAGS = $(inherited) -D COCOAPODS 8 | PODS_BUILD_DIR = ${BUILD_DIR} 9 | PODS_CONFIGURATION_BUILD_DIR = ${PODS_BUILD_DIR}/$(CONFIGURATION)$(EFFECTIVE_PLATFORM_NAME) 10 | PODS_ROOT = ${SRCROOT} 11 | PODS_TARGET_SRCROOT = ${PODS_ROOT}/FBSnapshotTestCase 12 | PRODUCT_BUNDLE_IDENTIFIER = org.cocoapods.${PRODUCT_NAME:rfc1034identifier} 13 | SKIP_INSTALL = YES 14 | SWIFT_INCLUDE_PATHS = $(inherited) "$(PLATFORM_DIR)/Developer/usr/lib" 15 | SYSTEM_FRAMEWORK_SEARCH_PATHS = $(inherited) "$(PLATFORM_DIR)/Developer/Library/Frameworks" 16 | USE_RECURSIVE_SCRIPT_INPUTS_IN_SCRIPT_PHASES = YES 17 | -------------------------------------------------------------------------------- /Example/Pods/Target Support Files/Pods-ZLBridge_Example/Pods-ZLBridge_Example-Info.plist: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | CFBundleDevelopmentRegion 6 | en 7 | CFBundleExecutable 8 | ${EXECUTABLE_NAME} 9 | CFBundleIdentifier 10 | ${PRODUCT_BUNDLE_IDENTIFIER} 11 | CFBundleInfoDictionaryVersion 12 | 6.0 13 | CFBundleName 14 | ${PRODUCT_NAME} 15 | CFBundlePackageType 16 | FMWK 17 | CFBundleShortVersionString 18 | 1.0.0 19 | CFBundleSignature 20 | ???? 21 | CFBundleVersion 22 | ${CURRENT_PROJECT_VERSION} 23 | NSPrincipalClass 24 | 25 | 26 | 27 | -------------------------------------------------------------------------------- /Example/Pods/Target Support Files/Pods-ZLBridge_Example/Pods-ZLBridge_Example-acknowledgements.markdown: -------------------------------------------------------------------------------- 1 | # Acknowledgements 2 | This application makes use of the following third party libraries: 3 | 4 | ## ZLBridge 5 | 6 | Copyright (c) 2021 范鹏 <2551412939@qq.com> 7 | 8 | Permission is hereby granted, free of charge, to any person obtaining a copy 9 | of this software and associated documentation files (the "Software"), to deal 10 | in the Software without restriction, including without limitation the rights 11 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 12 | copies of the Software, and to permit persons to whom the Software is 13 | furnished to do so, subject to the following conditions: 14 | 15 | The above copyright notice and this permission notice shall be included in 16 | all copies or substantial portions of the Software. 17 | 18 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 19 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 20 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 21 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 22 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 23 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN 24 | THE SOFTWARE. 25 | 26 | Generated by CocoaPods - https://cocoapods.org 27 | -------------------------------------------------------------------------------- /Example/Pods/Target Support Files/Pods-ZLBridge_Example/Pods-ZLBridge_Example-acknowledgements.plist: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | PreferenceSpecifiers 6 | 7 | 8 | FooterText 9 | This application makes use of the following third party libraries: 10 | Title 11 | Acknowledgements 12 | Type 13 | PSGroupSpecifier 14 | 15 | 16 | FooterText 17 | Copyright (c) 2021 范鹏 <2551412939@qq.com> 18 | 19 | Permission is hereby granted, free of charge, to any person obtaining a copy 20 | of this software and associated documentation files (the "Software"), to deal 21 | in the Software without restriction, including without limitation the rights 22 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 23 | copies of the Software, and to permit persons to whom the Software is 24 | furnished to do so, subject to the following conditions: 25 | 26 | The above copyright notice and this permission notice shall be included in 27 | all copies or substantial portions of the Software. 28 | 29 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 30 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 31 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 32 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 33 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 34 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN 35 | THE SOFTWARE. 36 | 37 | License 38 | MIT 39 | Title 40 | ZLBridge 41 | Type 42 | PSGroupSpecifier 43 | 44 | 45 | FooterText 46 | Generated by CocoaPods - https://cocoapods.org 47 | Title 48 | 49 | Type 50 | PSGroupSpecifier 51 | 52 | 53 | StringsTable 54 | Acknowledgements 55 | Title 56 | Acknowledgements 57 | 58 | 59 | -------------------------------------------------------------------------------- /Example/Pods/Target Support Files/Pods-ZLBridge_Example/Pods-ZLBridge_Example-dummy.m: -------------------------------------------------------------------------------- 1 | #import 2 | @interface PodsDummy_Pods_ZLBridge_Example : NSObject 3 | @end 4 | @implementation PodsDummy_Pods_ZLBridge_Example 5 | @end 6 | -------------------------------------------------------------------------------- /Example/Pods/Target Support Files/Pods-ZLBridge_Example/Pods-ZLBridge_Example-frameworks.sh: -------------------------------------------------------------------------------- 1 | #!/bin/sh 2 | set -e 3 | set -u 4 | set -o pipefail 5 | 6 | function on_error { 7 | echo "$(realpath -mq "${0}"):$1: error: Unexpected failure" 8 | } 9 | trap 'on_error $LINENO' ERR 10 | 11 | if [ -z ${FRAMEWORKS_FOLDER_PATH+x} ]; then 12 | # If FRAMEWORKS_FOLDER_PATH is not set, then there's nowhere for us to copy 13 | # frameworks to, so exit 0 (signalling the script phase was successful). 14 | exit 0 15 | fi 16 | 17 | echo "mkdir -p ${CONFIGURATION_BUILD_DIR}/${FRAMEWORKS_FOLDER_PATH}" 18 | mkdir -p "${CONFIGURATION_BUILD_DIR}/${FRAMEWORKS_FOLDER_PATH}" 19 | 20 | COCOAPODS_PARALLEL_CODE_SIGN="${COCOAPODS_PARALLEL_CODE_SIGN:-false}" 21 | SWIFT_STDLIB_PATH="${DT_TOOLCHAIN_DIR}/usr/lib/swift/${PLATFORM_NAME}" 22 | 23 | # Used as a return value for each invocation of `strip_invalid_archs` function. 24 | STRIP_BINARY_RETVAL=0 25 | 26 | # This protects against multiple targets copying the same framework dependency at the same time. The solution 27 | # was originally proposed here: https://lists.samba.org/archive/rsync/2008-February/020158.html 28 | RSYNC_PROTECT_TMP_FILES=(--filter "P .*.??????") 29 | 30 | # Copies and strips a vendored framework 31 | install_framework() 32 | { 33 | if [ -r "${BUILT_PRODUCTS_DIR}/$1" ]; then 34 | local source="${BUILT_PRODUCTS_DIR}/$1" 35 | elif [ -r "${BUILT_PRODUCTS_DIR}/$(basename "$1")" ]; then 36 | local source="${BUILT_PRODUCTS_DIR}/$(basename "$1")" 37 | elif [ -r "$1" ]; then 38 | local source="$1" 39 | fi 40 | 41 | local destination="${TARGET_BUILD_DIR}/${FRAMEWORKS_FOLDER_PATH}" 42 | 43 | if [ -L "${source}" ]; then 44 | echo "Symlinked..." 45 | source="$(readlink "${source}")" 46 | fi 47 | 48 | # Use filter instead of exclude so missing patterns don't throw errors. 49 | echo "rsync --delete -av "${RSYNC_PROTECT_TMP_FILES[@]}" --links --filter \"- CVS/\" --filter \"- .svn/\" --filter \"- .git/\" --filter \"- .hg/\" --filter \"- Headers\" --filter \"- PrivateHeaders\" --filter \"- Modules\" \"${source}\" \"${destination}\"" 50 | rsync --delete -av "${RSYNC_PROTECT_TMP_FILES[@]}" --links --filter "- CVS/" --filter "- .svn/" --filter "- .git/" --filter "- .hg/" --filter "- Headers" --filter "- PrivateHeaders" --filter "- Modules" "${source}" "${destination}" 51 | 52 | local basename 53 | basename="$(basename -s .framework "$1")" 54 | binary="${destination}/${basename}.framework/${basename}" 55 | 56 | if ! [ -r "$binary" ]; then 57 | binary="${destination}/${basename}" 58 | elif [ -L "${binary}" ]; then 59 | echo "Destination binary is symlinked..." 60 | dirname="$(dirname "${binary}")" 61 | binary="${dirname}/$(readlink "${binary}")" 62 | fi 63 | 64 | # Strip invalid architectures so "fat" simulator / device frameworks work on device 65 | if [[ "$(file "$binary")" == *"dynamically linked shared library"* ]]; then 66 | strip_invalid_archs "$binary" 67 | fi 68 | 69 | # Resign the code if required by the build settings to avoid unstable apps 70 | code_sign_if_enabled "${destination}/$(basename "$1")" 71 | 72 | # Embed linked Swift runtime libraries. No longer necessary as of Xcode 7. 73 | if [ "${XCODE_VERSION_MAJOR}" -lt 7 ]; then 74 | local swift_runtime_libs 75 | swift_runtime_libs=$(xcrun otool -LX "$binary" | grep --color=never @rpath/libswift | sed -E s/@rpath\\/\(.+dylib\).*/\\1/g | uniq -u) 76 | for lib in $swift_runtime_libs; do 77 | echo "rsync -auv \"${SWIFT_STDLIB_PATH}/${lib}\" \"${destination}\"" 78 | rsync -auv "${SWIFT_STDLIB_PATH}/${lib}" "${destination}" 79 | code_sign_if_enabled "${destination}/${lib}" 80 | done 81 | fi 82 | } 83 | 84 | # Copies and strips a vendored dSYM 85 | install_dsym() { 86 | local source="$1" 87 | warn_missing_arch=${2:-true} 88 | if [ -r "$source" ]; then 89 | # Copy the dSYM into the targets temp dir. 90 | echo "rsync --delete -av "${RSYNC_PROTECT_TMP_FILES[@]}" --filter \"- CVS/\" --filter \"- .svn/\" --filter \"- .git/\" --filter \"- .hg/\" --filter \"- Headers\" --filter \"- PrivateHeaders\" --filter \"- Modules\" \"${source}\" \"${DERIVED_FILES_DIR}\"" 91 | rsync --delete -av "${RSYNC_PROTECT_TMP_FILES[@]}" --filter "- CVS/" --filter "- .svn/" --filter "- .git/" --filter "- .hg/" --filter "- Headers" --filter "- PrivateHeaders" --filter "- Modules" "${source}" "${DERIVED_FILES_DIR}" 92 | 93 | local basename 94 | basename="$(basename -s .dSYM "$source")" 95 | binary_name="$(ls "$source/Contents/Resources/DWARF")" 96 | binary="${DERIVED_FILES_DIR}/${basename}.dSYM/Contents/Resources/DWARF/${binary_name}" 97 | 98 | # Strip invalid architectures so "fat" simulator / device frameworks work on device 99 | if [[ "$(file "$binary")" == *"Mach-O "*"dSYM companion"* ]]; then 100 | strip_invalid_archs "$binary" "$warn_missing_arch" 101 | fi 102 | 103 | if [[ $STRIP_BINARY_RETVAL == 1 ]]; then 104 | # Move the stripped file into its final destination. 105 | echo "rsync --delete -av "${RSYNC_PROTECT_TMP_FILES[@]}" --links --filter \"- CVS/\" --filter \"- .svn/\" --filter \"- .git/\" --filter \"- .hg/\" --filter \"- Headers\" --filter \"- PrivateHeaders\" --filter \"- Modules\" \"${DERIVED_FILES_DIR}/${basename}.framework.dSYM\" \"${DWARF_DSYM_FOLDER_PATH}\"" 106 | rsync --delete -av "${RSYNC_PROTECT_TMP_FILES[@]}" --links --filter "- CVS/" --filter "- .svn/" --filter "- .git/" --filter "- .hg/" --filter "- Headers" --filter "- PrivateHeaders" --filter "- Modules" "${DERIVED_FILES_DIR}/${basename}.dSYM" "${DWARF_DSYM_FOLDER_PATH}" 107 | else 108 | # The dSYM was not stripped at all, in this case touch a fake folder so the input/output paths from Xcode do not reexecute this script because the file is missing. 109 | touch "${DWARF_DSYM_FOLDER_PATH}/${basename}.dSYM" 110 | fi 111 | fi 112 | } 113 | 114 | # Copies the bcsymbolmap files of a vendored framework 115 | install_bcsymbolmap() { 116 | local bcsymbolmap_path="$1" 117 | local destination="${BUILT_PRODUCTS_DIR}" 118 | echo "rsync --delete -av "${RSYNC_PROTECT_TMP_FILES[@]}" --filter "- CVS/" --filter "- .svn/" --filter "- .git/" --filter "- .hg/" --filter "- Headers" --filter "- PrivateHeaders" --filter "- Modules" "${bcsymbolmap_path}" "${destination}"" 119 | rsync --delete -av "${RSYNC_PROTECT_TMP_FILES[@]}" --filter "- CVS/" --filter "- .svn/" --filter "- .git/" --filter "- .hg/" --filter "- Headers" --filter "- PrivateHeaders" --filter "- Modules" "${bcsymbolmap_path}" "${destination}" 120 | } 121 | 122 | # Signs a framework with the provided identity 123 | code_sign_if_enabled() { 124 | if [ -n "${EXPANDED_CODE_SIGN_IDENTITY:-}" -a "${CODE_SIGNING_REQUIRED:-}" != "NO" -a "${CODE_SIGNING_ALLOWED}" != "NO" ]; then 125 | # Use the current code_sign_identity 126 | echo "Code Signing $1 with Identity ${EXPANDED_CODE_SIGN_IDENTITY_NAME}" 127 | local code_sign_cmd="/usr/bin/codesign --force --sign ${EXPANDED_CODE_SIGN_IDENTITY} ${OTHER_CODE_SIGN_FLAGS:-} --preserve-metadata=identifier,entitlements '$1'" 128 | 129 | if [ "${COCOAPODS_PARALLEL_CODE_SIGN}" == "true" ]; then 130 | code_sign_cmd="$code_sign_cmd &" 131 | fi 132 | echo "$code_sign_cmd" 133 | eval "$code_sign_cmd" 134 | fi 135 | } 136 | 137 | # Strip invalid architectures 138 | strip_invalid_archs() { 139 | binary="$1" 140 | warn_missing_arch=${2:-true} 141 | # Get architectures for current target binary 142 | binary_archs="$(lipo -info "$binary" | rev | cut -d ':' -f1 | awk '{$1=$1;print}' | rev)" 143 | # Intersect them with the architectures we are building for 144 | intersected_archs="$(echo ${ARCHS[@]} ${binary_archs[@]} | tr ' ' '\n' | sort | uniq -d)" 145 | # If there are no archs supported by this binary then warn the user 146 | if [[ -z "$intersected_archs" ]]; then 147 | if [[ "$warn_missing_arch" == "true" ]]; then 148 | echo "warning: [CP] Vendored binary '$binary' contains architectures ($binary_archs) none of which match the current build architectures ($ARCHS)." 149 | fi 150 | STRIP_BINARY_RETVAL=0 151 | return 152 | fi 153 | stripped="" 154 | for arch in $binary_archs; do 155 | if ! [[ "${ARCHS}" == *"$arch"* ]]; then 156 | # Strip non-valid architectures in-place 157 | lipo -remove "$arch" -output "$binary" "$binary" 158 | stripped="$stripped $arch" 159 | fi 160 | done 161 | if [[ "$stripped" ]]; then 162 | echo "Stripped $binary of architectures:$stripped" 163 | fi 164 | STRIP_BINARY_RETVAL=1 165 | } 166 | 167 | install_artifact() { 168 | artifact="$1" 169 | base="$(basename "$artifact")" 170 | case $base in 171 | *.framework) 172 | install_framework "$artifact" 173 | ;; 174 | *.dSYM) 175 | # Suppress arch warnings since XCFrameworks will include many dSYM files 176 | install_dsym "$artifact" "false" 177 | ;; 178 | *.bcsymbolmap) 179 | install_bcsymbolmap "$artifact" 180 | ;; 181 | *) 182 | echo "error: Unrecognized artifact "$artifact"" 183 | ;; 184 | esac 185 | } 186 | 187 | copy_artifacts() { 188 | file_list="$1" 189 | while read artifact; do 190 | install_artifact "$artifact" 191 | done <$file_list 192 | } 193 | 194 | ARTIFACT_LIST_FILE="${BUILT_PRODUCTS_DIR}/cocoapods-artifacts-${CONFIGURATION}.txt" 195 | if [ -r "${ARTIFACT_LIST_FILE}" ]; then 196 | copy_artifacts "${ARTIFACT_LIST_FILE}" 197 | fi 198 | 199 | if [[ "$CONFIGURATION" == "Debug" ]]; then 200 | install_framework "${BUILT_PRODUCTS_DIR}/ZLBridge/ZLBridge.framework" 201 | fi 202 | if [[ "$CONFIGURATION" == "Release" ]]; then 203 | install_framework "${BUILT_PRODUCTS_DIR}/ZLBridge/ZLBridge.framework" 204 | fi 205 | if [ "${COCOAPODS_PARALLEL_CODE_SIGN}" == "true" ]; then 206 | wait 207 | fi 208 | -------------------------------------------------------------------------------- /Example/Pods/Target Support Files/Pods-ZLBridge_Example/Pods-ZLBridge_Example-umbrella.h: -------------------------------------------------------------------------------- 1 | #ifdef __OBJC__ 2 | #import 3 | #else 4 | #ifndef FOUNDATION_EXPORT 5 | #if defined(__cplusplus) 6 | #define FOUNDATION_EXPORT extern "C" 7 | #else 8 | #define FOUNDATION_EXPORT extern 9 | #endif 10 | #endif 11 | #endif 12 | 13 | 14 | FOUNDATION_EXPORT double Pods_ZLBridge_ExampleVersionNumber; 15 | FOUNDATION_EXPORT const unsigned char Pods_ZLBridge_ExampleVersionString[]; 16 | 17 | -------------------------------------------------------------------------------- /Example/Pods/Target Support Files/Pods-ZLBridge_Example/Pods-ZLBridge_Example.debug.xcconfig: -------------------------------------------------------------------------------- 1 | FRAMEWORK_SEARCH_PATHS = $(inherited) "${PODS_CONFIGURATION_BUILD_DIR}/ZLBridge" 2 | GCC_PREPROCESSOR_DEFINITIONS = $(inherited) COCOAPODS=1 3 | HEADER_SEARCH_PATHS = $(inherited) "${PODS_CONFIGURATION_BUILD_DIR}/ZLBridge/ZLBridge.framework/Headers" 4 | LD_RUNPATH_SEARCH_PATHS = $(inherited) '@executable_path/Frameworks' '@loader_path/Frameworks' 5 | OTHER_LDFLAGS = $(inherited) -framework "ZLBridge" 6 | PODS_BUILD_DIR = ${BUILD_DIR} 7 | PODS_CONFIGURATION_BUILD_DIR = ${PODS_BUILD_DIR}/$(CONFIGURATION)$(EFFECTIVE_PLATFORM_NAME) 8 | PODS_PODFILE_DIR_PATH = ${SRCROOT}/. 9 | PODS_ROOT = ${SRCROOT}/Pods 10 | USE_RECURSIVE_SCRIPT_INPUTS_IN_SCRIPT_PHASES = YES 11 | -------------------------------------------------------------------------------- /Example/Pods/Target Support Files/Pods-ZLBridge_Example/Pods-ZLBridge_Example.modulemap: -------------------------------------------------------------------------------- 1 | framework module Pods_ZLBridge_Example { 2 | umbrella header "Pods-ZLBridge_Example-umbrella.h" 3 | 4 | export * 5 | module * { export * } 6 | } 7 | -------------------------------------------------------------------------------- /Example/Pods/Target Support Files/Pods-ZLBridge_Example/Pods-ZLBridge_Example.release.xcconfig: -------------------------------------------------------------------------------- 1 | FRAMEWORK_SEARCH_PATHS = $(inherited) "${PODS_CONFIGURATION_BUILD_DIR}/ZLBridge" 2 | GCC_PREPROCESSOR_DEFINITIONS = $(inherited) COCOAPODS=1 3 | HEADER_SEARCH_PATHS = $(inherited) "${PODS_CONFIGURATION_BUILD_DIR}/ZLBridge/ZLBridge.framework/Headers" 4 | LD_RUNPATH_SEARCH_PATHS = $(inherited) '@executable_path/Frameworks' '@loader_path/Frameworks' 5 | OTHER_LDFLAGS = $(inherited) -framework "ZLBridge" 6 | PODS_BUILD_DIR = ${BUILD_DIR} 7 | PODS_CONFIGURATION_BUILD_DIR = ${PODS_BUILD_DIR}/$(CONFIGURATION)$(EFFECTIVE_PLATFORM_NAME) 8 | PODS_PODFILE_DIR_PATH = ${SRCROOT}/. 9 | PODS_ROOT = ${SRCROOT}/Pods 10 | USE_RECURSIVE_SCRIPT_INPUTS_IN_SCRIPT_PHASES = YES 11 | -------------------------------------------------------------------------------- /Example/Pods/Target Support Files/Pods-ZLBridge_Tests/Pods-ZLBridge_Tests-Info.plist: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | CFBundleDevelopmentRegion 6 | en 7 | CFBundleExecutable 8 | ${EXECUTABLE_NAME} 9 | CFBundleIdentifier 10 | ${PRODUCT_BUNDLE_IDENTIFIER} 11 | CFBundleInfoDictionaryVersion 12 | 6.0 13 | CFBundleName 14 | ${PRODUCT_NAME} 15 | CFBundlePackageType 16 | FMWK 17 | CFBundleShortVersionString 18 | 1.0.0 19 | CFBundleSignature 20 | ???? 21 | CFBundleVersion 22 | ${CURRENT_PROJECT_VERSION} 23 | NSPrincipalClass 24 | 25 | 26 | 27 | -------------------------------------------------------------------------------- /Example/Pods/Target Support Files/Pods-ZLBridge_Tests/Pods-ZLBridge_Tests-acknowledgements.markdown: -------------------------------------------------------------------------------- 1 | # Acknowledgements 2 | This application makes use of the following third party libraries: 3 | 4 | ## FBSnapshotTestCase 5 | 6 | BSD License 7 | 8 | For the FBSnapshotTestCase software 9 | 10 | Copyright (c) 2013, Facebook, Inc. 11 | All rights reserved. 12 | 13 | Redistribution and use in source and binary forms, with or without 14 | modification, are permitted provided that the following conditions are met: 15 | 16 | * Redistributions of source code must retain the above copyright notice, 17 | this list of conditions and the following disclaimer. 18 | * Redistributions in binary form must reproduce the above copyright notice, 19 | this list of conditions and the following disclaimer in the documentation 20 | and/or other materials provided with the distribution. 21 | * Neither the name Facebook nor the names of its contributors may be used to 22 | endorse or promote products derived from this software without specific 23 | prior written permission. 24 | 25 | THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" 26 | AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 27 | IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE 28 | DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE 29 | FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 30 | DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR 31 | SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER 32 | CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, 33 | OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE 34 | OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 35 | 36 | Generated by CocoaPods - https://cocoapods.org 37 | -------------------------------------------------------------------------------- /Example/Pods/Target Support Files/Pods-ZLBridge_Tests/Pods-ZLBridge_Tests-acknowledgements.plist: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | PreferenceSpecifiers 6 | 7 | 8 | FooterText 9 | This application makes use of the following third party libraries: 10 | Title 11 | Acknowledgements 12 | Type 13 | PSGroupSpecifier 14 | 15 | 16 | FooterText 17 | BSD License 18 | 19 | For the FBSnapshotTestCase software 20 | 21 | Copyright (c) 2013, Facebook, Inc. 22 | All rights reserved. 23 | 24 | Redistribution and use in source and binary forms, with or without 25 | modification, are permitted provided that the following conditions are met: 26 | 27 | * Redistributions of source code must retain the above copyright notice, 28 | this list of conditions and the following disclaimer. 29 | * Redistributions in binary form must reproduce the above copyright notice, 30 | this list of conditions and the following disclaimer in the documentation 31 | and/or other materials provided with the distribution. 32 | * Neither the name Facebook nor the names of its contributors may be used to 33 | endorse or promote products derived from this software without specific 34 | prior written permission. 35 | 36 | THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" 37 | AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 38 | IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE 39 | DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE 40 | FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 41 | DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR 42 | SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER 43 | CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, 44 | OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE 45 | OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 46 | 47 | License 48 | BSD 49 | Title 50 | FBSnapshotTestCase 51 | Type 52 | PSGroupSpecifier 53 | 54 | 55 | FooterText 56 | Generated by CocoaPods - https://cocoapods.org 57 | Title 58 | 59 | Type 60 | PSGroupSpecifier 61 | 62 | 63 | StringsTable 64 | Acknowledgements 65 | Title 66 | Acknowledgements 67 | 68 | 69 | -------------------------------------------------------------------------------- /Example/Pods/Target Support Files/Pods-ZLBridge_Tests/Pods-ZLBridge_Tests-dummy.m: -------------------------------------------------------------------------------- 1 | #import 2 | @interface PodsDummy_Pods_ZLBridge_Tests : NSObject 3 | @end 4 | @implementation PodsDummy_Pods_ZLBridge_Tests 5 | @end 6 | -------------------------------------------------------------------------------- /Example/Pods/Target Support Files/Pods-ZLBridge_Tests/Pods-ZLBridge_Tests-frameworks.sh: -------------------------------------------------------------------------------- 1 | #!/bin/sh 2 | set -e 3 | set -u 4 | set -o pipefail 5 | 6 | function on_error { 7 | echo "$(realpath -mq "${0}"):$1: error: Unexpected failure" 8 | } 9 | trap 'on_error $LINENO' ERR 10 | 11 | if [ -z ${FRAMEWORKS_FOLDER_PATH+x} ]; then 12 | # If FRAMEWORKS_FOLDER_PATH is not set, then there's nowhere for us to copy 13 | # frameworks to, so exit 0 (signalling the script phase was successful). 14 | exit 0 15 | fi 16 | 17 | echo "mkdir -p ${CONFIGURATION_BUILD_DIR}/${FRAMEWORKS_FOLDER_PATH}" 18 | mkdir -p "${CONFIGURATION_BUILD_DIR}/${FRAMEWORKS_FOLDER_PATH}" 19 | 20 | COCOAPODS_PARALLEL_CODE_SIGN="${COCOAPODS_PARALLEL_CODE_SIGN:-false}" 21 | SWIFT_STDLIB_PATH="${DT_TOOLCHAIN_DIR}/usr/lib/swift/${PLATFORM_NAME}" 22 | 23 | # Used as a return value for each invocation of `strip_invalid_archs` function. 24 | STRIP_BINARY_RETVAL=0 25 | 26 | # This protects against multiple targets copying the same framework dependency at the same time. The solution 27 | # was originally proposed here: https://lists.samba.org/archive/rsync/2008-February/020158.html 28 | RSYNC_PROTECT_TMP_FILES=(--filter "P .*.??????") 29 | 30 | # Copies and strips a vendored framework 31 | install_framework() 32 | { 33 | if [ -r "${BUILT_PRODUCTS_DIR}/$1" ]; then 34 | local source="${BUILT_PRODUCTS_DIR}/$1" 35 | elif [ -r "${BUILT_PRODUCTS_DIR}/$(basename "$1")" ]; then 36 | local source="${BUILT_PRODUCTS_DIR}/$(basename "$1")" 37 | elif [ -r "$1" ]; then 38 | local source="$1" 39 | fi 40 | 41 | local destination="${TARGET_BUILD_DIR}/${FRAMEWORKS_FOLDER_PATH}" 42 | 43 | if [ -L "${source}" ]; then 44 | echo "Symlinked..." 45 | source="$(readlink "${source}")" 46 | fi 47 | 48 | # Use filter instead of exclude so missing patterns don't throw errors. 49 | echo "rsync --delete -av "${RSYNC_PROTECT_TMP_FILES[@]}" --links --filter \"- CVS/\" --filter \"- .svn/\" --filter \"- .git/\" --filter \"- .hg/\" --filter \"- Headers\" --filter \"- PrivateHeaders\" --filter \"- Modules\" \"${source}\" \"${destination}\"" 50 | rsync --delete -av "${RSYNC_PROTECT_TMP_FILES[@]}" --links --filter "- CVS/" --filter "- .svn/" --filter "- .git/" --filter "- .hg/" --filter "- Headers" --filter "- PrivateHeaders" --filter "- Modules" "${source}" "${destination}" 51 | 52 | local basename 53 | basename="$(basename -s .framework "$1")" 54 | binary="${destination}/${basename}.framework/${basename}" 55 | 56 | if ! [ -r "$binary" ]; then 57 | binary="${destination}/${basename}" 58 | elif [ -L "${binary}" ]; then 59 | echo "Destination binary is symlinked..." 60 | dirname="$(dirname "${binary}")" 61 | binary="${dirname}/$(readlink "${binary}")" 62 | fi 63 | 64 | # Strip invalid architectures so "fat" simulator / device frameworks work on device 65 | if [[ "$(file "$binary")" == *"dynamically linked shared library"* ]]; then 66 | strip_invalid_archs "$binary" 67 | fi 68 | 69 | # Resign the code if required by the build settings to avoid unstable apps 70 | code_sign_if_enabled "${destination}/$(basename "$1")" 71 | 72 | # Embed linked Swift runtime libraries. No longer necessary as of Xcode 7. 73 | if [ "${XCODE_VERSION_MAJOR}" -lt 7 ]; then 74 | local swift_runtime_libs 75 | swift_runtime_libs=$(xcrun otool -LX "$binary" | grep --color=never @rpath/libswift | sed -E s/@rpath\\/\(.+dylib\).*/\\1/g | uniq -u) 76 | for lib in $swift_runtime_libs; do 77 | echo "rsync -auv \"${SWIFT_STDLIB_PATH}/${lib}\" \"${destination}\"" 78 | rsync -auv "${SWIFT_STDLIB_PATH}/${lib}" "${destination}" 79 | code_sign_if_enabled "${destination}/${lib}" 80 | done 81 | fi 82 | } 83 | 84 | # Copies and strips a vendored dSYM 85 | install_dsym() { 86 | local source="$1" 87 | warn_missing_arch=${2:-true} 88 | if [ -r "$source" ]; then 89 | # Copy the dSYM into the targets temp dir. 90 | echo "rsync --delete -av "${RSYNC_PROTECT_TMP_FILES[@]}" --filter \"- CVS/\" --filter \"- .svn/\" --filter \"- .git/\" --filter \"- .hg/\" --filter \"- Headers\" --filter \"- PrivateHeaders\" --filter \"- Modules\" \"${source}\" \"${DERIVED_FILES_DIR}\"" 91 | rsync --delete -av "${RSYNC_PROTECT_TMP_FILES[@]}" --filter "- CVS/" --filter "- .svn/" --filter "- .git/" --filter "- .hg/" --filter "- Headers" --filter "- PrivateHeaders" --filter "- Modules" "${source}" "${DERIVED_FILES_DIR}" 92 | 93 | local basename 94 | basename="$(basename -s .dSYM "$source")" 95 | binary_name="$(ls "$source/Contents/Resources/DWARF")" 96 | binary="${DERIVED_FILES_DIR}/${basename}.dSYM/Contents/Resources/DWARF/${binary_name}" 97 | 98 | # Strip invalid architectures so "fat" simulator / device frameworks work on device 99 | if [[ "$(file "$binary")" == *"Mach-O "*"dSYM companion"* ]]; then 100 | strip_invalid_archs "$binary" "$warn_missing_arch" 101 | fi 102 | 103 | if [[ $STRIP_BINARY_RETVAL == 1 ]]; then 104 | # Move the stripped file into its final destination. 105 | echo "rsync --delete -av "${RSYNC_PROTECT_TMP_FILES[@]}" --links --filter \"- CVS/\" --filter \"- .svn/\" --filter \"- .git/\" --filter \"- .hg/\" --filter \"- Headers\" --filter \"- PrivateHeaders\" --filter \"- Modules\" \"${DERIVED_FILES_DIR}/${basename}.framework.dSYM\" \"${DWARF_DSYM_FOLDER_PATH}\"" 106 | rsync --delete -av "${RSYNC_PROTECT_TMP_FILES[@]}" --links --filter "- CVS/" --filter "- .svn/" --filter "- .git/" --filter "- .hg/" --filter "- Headers" --filter "- PrivateHeaders" --filter "- Modules" "${DERIVED_FILES_DIR}/${basename}.dSYM" "${DWARF_DSYM_FOLDER_PATH}" 107 | else 108 | # The dSYM was not stripped at all, in this case touch a fake folder so the input/output paths from Xcode do not reexecute this script because the file is missing. 109 | touch "${DWARF_DSYM_FOLDER_PATH}/${basename}.dSYM" 110 | fi 111 | fi 112 | } 113 | 114 | # Copies the bcsymbolmap files of a vendored framework 115 | install_bcsymbolmap() { 116 | local bcsymbolmap_path="$1" 117 | local destination="${BUILT_PRODUCTS_DIR}" 118 | echo "rsync --delete -av "${RSYNC_PROTECT_TMP_FILES[@]}" --filter "- CVS/" --filter "- .svn/" --filter "- .git/" --filter "- .hg/" --filter "- Headers" --filter "- PrivateHeaders" --filter "- Modules" "${bcsymbolmap_path}" "${destination}"" 119 | rsync --delete -av "${RSYNC_PROTECT_TMP_FILES[@]}" --filter "- CVS/" --filter "- .svn/" --filter "- .git/" --filter "- .hg/" --filter "- Headers" --filter "- PrivateHeaders" --filter "- Modules" "${bcsymbolmap_path}" "${destination}" 120 | } 121 | 122 | # Signs a framework with the provided identity 123 | code_sign_if_enabled() { 124 | if [ -n "${EXPANDED_CODE_SIGN_IDENTITY:-}" -a "${CODE_SIGNING_REQUIRED:-}" != "NO" -a "${CODE_SIGNING_ALLOWED}" != "NO" ]; then 125 | # Use the current code_sign_identity 126 | echo "Code Signing $1 with Identity ${EXPANDED_CODE_SIGN_IDENTITY_NAME}" 127 | local code_sign_cmd="/usr/bin/codesign --force --sign ${EXPANDED_CODE_SIGN_IDENTITY} ${OTHER_CODE_SIGN_FLAGS:-} --preserve-metadata=identifier,entitlements '$1'" 128 | 129 | if [ "${COCOAPODS_PARALLEL_CODE_SIGN}" == "true" ]; then 130 | code_sign_cmd="$code_sign_cmd &" 131 | fi 132 | echo "$code_sign_cmd" 133 | eval "$code_sign_cmd" 134 | fi 135 | } 136 | 137 | # Strip invalid architectures 138 | strip_invalid_archs() { 139 | binary="$1" 140 | warn_missing_arch=${2:-true} 141 | # Get architectures for current target binary 142 | binary_archs="$(lipo -info "$binary" | rev | cut -d ':' -f1 | awk '{$1=$1;print}' | rev)" 143 | # Intersect them with the architectures we are building for 144 | intersected_archs="$(echo ${ARCHS[@]} ${binary_archs[@]} | tr ' ' '\n' | sort | uniq -d)" 145 | # If there are no archs supported by this binary then warn the user 146 | if [[ -z "$intersected_archs" ]]; then 147 | if [[ "$warn_missing_arch" == "true" ]]; then 148 | echo "warning: [CP] Vendored binary '$binary' contains architectures ($binary_archs) none of which match the current build architectures ($ARCHS)." 149 | fi 150 | STRIP_BINARY_RETVAL=0 151 | return 152 | fi 153 | stripped="" 154 | for arch in $binary_archs; do 155 | if ! [[ "${ARCHS}" == *"$arch"* ]]; then 156 | # Strip non-valid architectures in-place 157 | lipo -remove "$arch" -output "$binary" "$binary" 158 | stripped="$stripped $arch" 159 | fi 160 | done 161 | if [[ "$stripped" ]]; then 162 | echo "Stripped $binary of architectures:$stripped" 163 | fi 164 | STRIP_BINARY_RETVAL=1 165 | } 166 | 167 | install_artifact() { 168 | artifact="$1" 169 | base="$(basename "$artifact")" 170 | case $base in 171 | *.framework) 172 | install_framework "$artifact" 173 | ;; 174 | *.dSYM) 175 | # Suppress arch warnings since XCFrameworks will include many dSYM files 176 | install_dsym "$artifact" "false" 177 | ;; 178 | *.bcsymbolmap) 179 | install_bcsymbolmap "$artifact" 180 | ;; 181 | *) 182 | echo "error: Unrecognized artifact "$artifact"" 183 | ;; 184 | esac 185 | } 186 | 187 | copy_artifacts() { 188 | file_list="$1" 189 | while read artifact; do 190 | install_artifact "$artifact" 191 | done <$file_list 192 | } 193 | 194 | ARTIFACT_LIST_FILE="${BUILT_PRODUCTS_DIR}/cocoapods-artifacts-${CONFIGURATION}.txt" 195 | if [ -r "${ARTIFACT_LIST_FILE}" ]; then 196 | copy_artifacts "${ARTIFACT_LIST_FILE}" 197 | fi 198 | 199 | if [[ "$CONFIGURATION" == "Debug" ]]; then 200 | install_framework "${BUILT_PRODUCTS_DIR}/FBSnapshotTestCase/FBSnapshotTestCase.framework" 201 | fi 202 | if [[ "$CONFIGURATION" == "Release" ]]; then 203 | install_framework "${BUILT_PRODUCTS_DIR}/FBSnapshotTestCase/FBSnapshotTestCase.framework" 204 | fi 205 | if [ "${COCOAPODS_PARALLEL_CODE_SIGN}" == "true" ]; then 206 | wait 207 | fi 208 | -------------------------------------------------------------------------------- /Example/Pods/Target Support Files/Pods-ZLBridge_Tests/Pods-ZLBridge_Tests-umbrella.h: -------------------------------------------------------------------------------- 1 | #ifdef __OBJC__ 2 | #import 3 | #else 4 | #ifndef FOUNDATION_EXPORT 5 | #if defined(__cplusplus) 6 | #define FOUNDATION_EXPORT extern "C" 7 | #else 8 | #define FOUNDATION_EXPORT extern 9 | #endif 10 | #endif 11 | #endif 12 | 13 | 14 | FOUNDATION_EXPORT double Pods_ZLBridge_TestsVersionNumber; 15 | FOUNDATION_EXPORT const unsigned char Pods_ZLBridge_TestsVersionString[]; 16 | 17 | -------------------------------------------------------------------------------- /Example/Pods/Target Support Files/Pods-ZLBridge_Tests/Pods-ZLBridge_Tests.debug.xcconfig: -------------------------------------------------------------------------------- 1 | ALWAYS_EMBED_SWIFT_STANDARD_LIBRARIES = YES 2 | FRAMEWORK_SEARCH_PATHS = $(inherited) "$(PLATFORM_DIR)/Developer/Library/Frameworks" "${PODS_CONFIGURATION_BUILD_DIR}/FBSnapshotTestCase" "${PODS_CONFIGURATION_BUILD_DIR}/ZLBridge" 3 | GCC_PREPROCESSOR_DEFINITIONS = $(inherited) COCOAPODS=1 4 | HEADER_SEARCH_PATHS = $(inherited) "${PODS_CONFIGURATION_BUILD_DIR}/FBSnapshotTestCase/FBSnapshotTestCase.framework/Headers" "${PODS_CONFIGURATION_BUILD_DIR}/ZLBridge/ZLBridge.framework/Headers" 5 | LD_RUNPATH_SEARCH_PATHS = $(inherited) '@executable_path/Frameworks' '@loader_path/Frameworks' 6 | OTHER_LDFLAGS = $(inherited) -framework "FBSnapshotTestCase" -framework "Foundation" -framework "QuartzCore" -framework "UIKit" -framework "XCTest" -framework "ZLBridge" 7 | OTHER_SWIFT_FLAGS = $(inherited) -D COCOAPODS 8 | PODS_BUILD_DIR = ${BUILD_DIR} 9 | PODS_CONFIGURATION_BUILD_DIR = ${PODS_BUILD_DIR}/$(CONFIGURATION)$(EFFECTIVE_PLATFORM_NAME) 10 | PODS_PODFILE_DIR_PATH = ${SRCROOT}/. 11 | PODS_ROOT = ${SRCROOT}/Pods 12 | USE_RECURSIVE_SCRIPT_INPUTS_IN_SCRIPT_PHASES = YES 13 | -------------------------------------------------------------------------------- /Example/Pods/Target Support Files/Pods-ZLBridge_Tests/Pods-ZLBridge_Tests.modulemap: -------------------------------------------------------------------------------- 1 | framework module Pods_ZLBridge_Tests { 2 | umbrella header "Pods-ZLBridge_Tests-umbrella.h" 3 | 4 | export * 5 | module * { export * } 6 | } 7 | -------------------------------------------------------------------------------- /Example/Pods/Target Support Files/Pods-ZLBridge_Tests/Pods-ZLBridge_Tests.release.xcconfig: -------------------------------------------------------------------------------- 1 | ALWAYS_EMBED_SWIFT_STANDARD_LIBRARIES = YES 2 | FRAMEWORK_SEARCH_PATHS = $(inherited) "$(PLATFORM_DIR)/Developer/Library/Frameworks" "${PODS_CONFIGURATION_BUILD_DIR}/FBSnapshotTestCase" "${PODS_CONFIGURATION_BUILD_DIR}/ZLBridge" 3 | GCC_PREPROCESSOR_DEFINITIONS = $(inherited) COCOAPODS=1 4 | HEADER_SEARCH_PATHS = $(inherited) "${PODS_CONFIGURATION_BUILD_DIR}/FBSnapshotTestCase/FBSnapshotTestCase.framework/Headers" "${PODS_CONFIGURATION_BUILD_DIR}/ZLBridge/ZLBridge.framework/Headers" 5 | LD_RUNPATH_SEARCH_PATHS = $(inherited) '@executable_path/Frameworks' '@loader_path/Frameworks' 6 | OTHER_LDFLAGS = $(inherited) -framework "FBSnapshotTestCase" -framework "Foundation" -framework "QuartzCore" -framework "UIKit" -framework "XCTest" -framework "ZLBridge" 7 | OTHER_SWIFT_FLAGS = $(inherited) -D COCOAPODS 8 | PODS_BUILD_DIR = ${BUILD_DIR} 9 | PODS_CONFIGURATION_BUILD_DIR = ${PODS_BUILD_DIR}/$(CONFIGURATION)$(EFFECTIVE_PLATFORM_NAME) 10 | PODS_PODFILE_DIR_PATH = ${SRCROOT}/. 11 | PODS_ROOT = ${SRCROOT}/Pods 12 | USE_RECURSIVE_SCRIPT_INPUTS_IN_SCRIPT_PHASES = YES 13 | -------------------------------------------------------------------------------- /Example/Pods/Target Support Files/ZLBridge/ResourceBundle-ZLBridge-ZLBridge-Info.plist: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | CFBundleDevelopmentRegion 6 | en 7 | CFBundleIdentifier 8 | ${PRODUCT_BUNDLE_IDENTIFIER} 9 | CFBundleInfoDictionaryVersion 10 | 6.0 11 | CFBundleName 12 | ${PRODUCT_NAME} 13 | CFBundlePackageType 14 | BNDL 15 | CFBundleShortVersionString 16 | 1.0.0 17 | CFBundleSignature 18 | ???? 19 | CFBundleVersion 20 | 1 21 | NSPrincipalClass 22 | 23 | 24 | 25 | -------------------------------------------------------------------------------- /Example/Pods/Target Support Files/ZLBridge/ZLBridge-Info.plist: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | CFBundleDevelopmentRegion 6 | en 7 | CFBundleExecutable 8 | ${EXECUTABLE_NAME} 9 | CFBundleIdentifier 10 | ${PRODUCT_BUNDLE_IDENTIFIER} 11 | CFBundleInfoDictionaryVersion 12 | 6.0 13 | CFBundleName 14 | ${PRODUCT_NAME} 15 | CFBundlePackageType 16 | FMWK 17 | CFBundleShortVersionString 18 | 1.0.0 19 | CFBundleSignature 20 | ???? 21 | CFBundleVersion 22 | ${CURRENT_PROJECT_VERSION} 23 | NSPrincipalClass 24 | 25 | 26 | 27 | -------------------------------------------------------------------------------- /Example/Pods/Target Support Files/ZLBridge/ZLBridge-dummy.m: -------------------------------------------------------------------------------- 1 | #import 2 | @interface PodsDummy_ZLBridge : NSObject 3 | @end 4 | @implementation PodsDummy_ZLBridge 5 | @end 6 | -------------------------------------------------------------------------------- /Example/Pods/Target Support Files/ZLBridge/ZLBridge-prefix.pch: -------------------------------------------------------------------------------- 1 | #ifdef __OBJC__ 2 | #import 3 | #else 4 | #ifndef FOUNDATION_EXPORT 5 | #if defined(__cplusplus) 6 | #define FOUNDATION_EXPORT extern "C" 7 | #else 8 | #define FOUNDATION_EXPORT extern 9 | #endif 10 | #endif 11 | #endif 12 | 13 | -------------------------------------------------------------------------------- /Example/Pods/Target Support Files/ZLBridge/ZLBridge-umbrella.h: -------------------------------------------------------------------------------- 1 | #ifdef __OBJC__ 2 | #import 3 | #else 4 | #ifndef FOUNDATION_EXPORT 5 | #if defined(__cplusplus) 6 | #define FOUNDATION_EXPORT extern "C" 7 | #else 8 | #define FOUNDATION_EXPORT extern 9 | #endif 10 | #endif 11 | #endif 12 | 13 | #import "WKWebView+ZLBridge.h" 14 | 15 | FOUNDATION_EXPORT double ZLBridgeVersionNumber; 16 | FOUNDATION_EXPORT const unsigned char ZLBridgeVersionString[]; 17 | 18 | -------------------------------------------------------------------------------- /Example/Pods/Target Support Files/ZLBridge/ZLBridge.debug.xcconfig: -------------------------------------------------------------------------------- 1 | CONFIGURATION_BUILD_DIR = ${PODS_CONFIGURATION_BUILD_DIR}/ZLBridge 2 | GCC_PREPROCESSOR_DEFINITIONS = $(inherited) COCOAPODS=1 3 | PODS_BUILD_DIR = ${BUILD_DIR} 4 | PODS_CONFIGURATION_BUILD_DIR = ${PODS_BUILD_DIR}/$(CONFIGURATION)$(EFFECTIVE_PLATFORM_NAME) 5 | PODS_ROOT = ${SRCROOT} 6 | PODS_TARGET_SRCROOT = ${PODS_ROOT}/../.. 7 | PRODUCT_BUNDLE_IDENTIFIER = org.cocoapods.${PRODUCT_NAME:rfc1034identifier} 8 | SKIP_INSTALL = YES 9 | USE_RECURSIVE_SCRIPT_INPUTS_IN_SCRIPT_PHASES = YES 10 | -------------------------------------------------------------------------------- /Example/Pods/Target Support Files/ZLBridge/ZLBridge.modulemap: -------------------------------------------------------------------------------- 1 | framework module ZLBridge { 2 | umbrella header "ZLBridge-umbrella.h" 3 | 4 | export * 5 | module * { export * } 6 | } 7 | -------------------------------------------------------------------------------- /Example/Pods/Target Support Files/ZLBridge/ZLBridge.release.xcconfig: -------------------------------------------------------------------------------- 1 | CONFIGURATION_BUILD_DIR = ${PODS_CONFIGURATION_BUILD_DIR}/ZLBridge 2 | GCC_PREPROCESSOR_DEFINITIONS = $(inherited) COCOAPODS=1 3 | PODS_BUILD_DIR = ${BUILD_DIR} 4 | PODS_CONFIGURATION_BUILD_DIR = ${PODS_BUILD_DIR}/$(CONFIGURATION)$(EFFECTIVE_PLATFORM_NAME) 5 | PODS_ROOT = ${SRCROOT} 6 | PODS_TARGET_SRCROOT = ${PODS_ROOT}/../.. 7 | PRODUCT_BUNDLE_IDENTIFIER = org.cocoapods.${PRODUCT_NAME:rfc1034identifier} 8 | SKIP_INSTALL = YES 9 | USE_RECURSIVE_SCRIPT_INPUTS_IN_SCRIPT_PHASES = YES 10 | -------------------------------------------------------------------------------- /Example/Tests/Tests-Info.plist: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | CFBundleDevelopmentRegion 6 | en 7 | CFBundleExecutable 8 | ${EXECUTABLE_NAME} 9 | CFBundleIdentifier 10 | $(PRODUCT_BUNDLE_IDENTIFIER) 11 | CFBundleInfoDictionaryVersion 12 | 6.0 13 | CFBundlePackageType 14 | BNDL 15 | CFBundleShortVersionString 16 | 1.0 17 | CFBundleSignature 18 | ???? 19 | CFBundleVersion 20 | 1 21 | 22 | 23 | -------------------------------------------------------------------------------- /Example/Tests/Tests-Prefix.pch: -------------------------------------------------------------------------------- 1 | // The contents of this file are implicitly included at the beginning of every test case source file. 2 | 3 | #ifdef __OBJC__ 4 | 5 | @import FBSnapshotTestCase; 6 | 7 | #endif 8 | -------------------------------------------------------------------------------- /Example/Tests/Tests.m: -------------------------------------------------------------------------------- 1 | // 2 | // ZLBridgeTests.m 3 | // ZLBridgeTests 4 | // 5 | // Created by 范鹏 on 06/04/2021. 6 | // Copyright (c) 2021 范鹏. All rights reserved. 7 | // 8 | 9 | @import XCTest; 10 | 11 | @interface Tests : XCTestCase 12 | 13 | @end 14 | 15 | @implementation Tests 16 | 17 | - (void)setUp 18 | { 19 | [super setUp]; 20 | // Put setup code here. This method is called before the invocation of each test method in the class. 21 | } 22 | 23 | - (void)tearDown 24 | { 25 | // Put teardown code here. This method is called after the invocation of each test method in the class. 26 | [super tearDown]; 27 | } 28 | 29 | - (void)testExample 30 | { 31 | XCTFail(@"No implementation for \"%s\"", __PRETTY_FUNCTION__); 32 | } 33 | 34 | @end 35 | 36 | -------------------------------------------------------------------------------- /Example/Tests/en.lproj/InfoPlist.strings: -------------------------------------------------------------------------------- 1 | /* Localized versions of Info.plist keys */ 2 | 3 | -------------------------------------------------------------------------------- /Example/ZLBridge.xcodeproj/project.xcworkspace/contents.xcworkspacedata: -------------------------------------------------------------------------------- 1 | 2 | 4 | 6 | 7 | 8 | -------------------------------------------------------------------------------- /Example/ZLBridge.xcodeproj/xcshareddata/xcschemes/ZLBridge-Example.xcscheme: -------------------------------------------------------------------------------- 1 | 2 | 5 | 8 | 9 | 15 | 21 | 22 | 23 | 24 | 25 | 30 | 31 | 33 | 39 | 40 | 41 | 42 | 43 | 49 | 50 | 51 | 52 | 53 | 54 | 64 | 66 | 72 | 73 | 74 | 75 | 76 | 77 | 83 | 85 | 91 | 92 | 93 | 94 | 96 | 97 | 100 | 101 | 102 | -------------------------------------------------------------------------------- /Example/ZLBridge.xcworkspace/contents.xcworkspacedata: -------------------------------------------------------------------------------- 1 | 2 | 4 | 6 | 7 | 9 | 10 | 11 | -------------------------------------------------------------------------------- /Example/ZLBridge.xcworkspace/xcuserdata/fanpeng.xcuserdatad/UserInterfaceState.xcuserstate: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/FPJack/ZLBridge-iOS/85cdbb62eeae25b02f2681d07456c0c501d1913b/Example/ZLBridge.xcworkspace/xcuserdata/fanpeng.xcuserdatad/UserInterfaceState.xcuserstate -------------------------------------------------------------------------------- /Example/ZLBridge.xcworkspace/xcuserdata/fanpeng.xcuserdatad/xcdebugger/Breakpoints_v2.xcbkptlist: -------------------------------------------------------------------------------- 1 | 2 | 6 | 7 | -------------------------------------------------------------------------------- /Example/ZLBridge/Base.lproj/LaunchScreen.storyboard: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15 | 16 | 17 | 18 | 19 | 20 | 21 | 22 | 23 | 24 | 25 | 26 | 27 | 28 | 29 | 30 | 31 | 32 | -------------------------------------------------------------------------------- /Example/ZLBridge/Base.lproj/Main.storyboard: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15 | 16 | 17 | 18 | 19 | 20 | 21 | 22 | 23 | 24 | 25 | 26 | 27 | 28 | 29 | 30 | 31 | 32 | 33 | 34 | 35 | 36 | 37 | 38 | 46 | 53 | 60 | 61 | 62 | 63 | 64 | 65 | 66 | 67 | 68 | 69 | 70 | 71 | 72 | 73 | 74 | 75 | 76 | 77 | -------------------------------------------------------------------------------- /Example/ZLBridge/Images.xcassets/AppIcon.appiconset/Contents.json: -------------------------------------------------------------------------------- 1 | { 2 | "images" : [ 3 | { 4 | "idiom" : "iphone", 5 | "size" : "20x20", 6 | "scale" : "2x" 7 | }, 8 | { 9 | "idiom" : "iphone", 10 | "size" : "20x20", 11 | "scale" : "3x" 12 | }, 13 | { 14 | "idiom" : "iphone", 15 | "size" : "29x29", 16 | "scale" : "2x" 17 | }, 18 | { 19 | "idiom" : "iphone", 20 | "size" : "29x29", 21 | "scale" : "3x" 22 | }, 23 | { 24 | "idiom" : "iphone", 25 | "size" : "40x40", 26 | "scale" : "2x" 27 | }, 28 | { 29 | "idiom" : "iphone", 30 | "size" : "40x40", 31 | "scale" : "3x" 32 | }, 33 | { 34 | "idiom" : "iphone", 35 | "size" : "60x60", 36 | "scale" : "2x" 37 | }, 38 | { 39 | "idiom" : "iphone", 40 | "size" : "60x60", 41 | "scale" : "3x" 42 | }, 43 | { 44 | "idiom" : "ipad", 45 | "size" : "20x20", 46 | "scale" : "1x" 47 | }, 48 | { 49 | "idiom" : "ipad", 50 | "size" : "20x20", 51 | "scale" : "2x" 52 | }, 53 | { 54 | "idiom" : "ipad", 55 | "size" : "29x29", 56 | "scale" : "1x" 57 | }, 58 | { 59 | "idiom" : "ipad", 60 | "size" : "29x29", 61 | "scale" : "2x" 62 | }, 63 | { 64 | "idiom" : "ipad", 65 | "size" : "40x40", 66 | "scale" : "1x" 67 | }, 68 | { 69 | "idiom" : "ipad", 70 | "size" : "40x40", 71 | "scale" : "2x" 72 | }, 73 | { 74 | "idiom" : "ipad", 75 | "size" : "76x76", 76 | "scale" : "1x" 77 | }, 78 | { 79 | "idiom" : "ipad", 80 | "size" : "76x76", 81 | "scale" : "2x" 82 | }, 83 | { 84 | "idiom" : "ipad", 85 | "size" : "83.5x83.5", 86 | "scale" : "2x" 87 | }, 88 | { 89 | "idiom" : "ios-marketing", 90 | "size" : "1024x1024", 91 | "scale" : "1x" 92 | } 93 | ], 94 | "info" : { 95 | "version" : 1, 96 | "author" : "xcode" 97 | } 98 | } 99 | -------------------------------------------------------------------------------- /Example/ZLBridge/ZLAppDelegate.h: -------------------------------------------------------------------------------- 1 | // 2 | // ZLAppDelegate.h 3 | // ZLBridge 4 | // 5 | // Created by 范鹏 on 06/04/2021. 6 | // Copyright (c) 2021 范鹏. All rights reserved. 7 | // 8 | 9 | @import UIKit; 10 | 11 | @interface ZLAppDelegate : UIResponder 12 | 13 | @property (strong, nonatomic) UIWindow *window; 14 | 15 | @end 16 | -------------------------------------------------------------------------------- /Example/ZLBridge/ZLAppDelegate.m: -------------------------------------------------------------------------------- 1 | // 2 | // ZLAppDelegate.m 3 | // ZLBridge 4 | // 5 | // Created by 范鹏 on 06/04/2021. 6 | // Copyright (c) 2021 范鹏. All rights reserved. 7 | // 8 | 9 | #import "ZLAppDelegate.h" 10 | 11 | @implementation ZLAppDelegate 12 | 13 | - (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions 14 | { 15 | // Override point for customization after application launch. 16 | return YES; 17 | } 18 | 19 | - (void)applicationWillResignActive:(UIApplication *)application 20 | { 21 | // Sent when the application is about to move from active to inactive state. This can occur for certain types of temporary interruptions (such as an incoming phone call or SMS message) or when the user quits the application and it begins the transition to the background state. 22 | // Use this method to pause ongoing tasks, disable timers, and throttle down OpenGL ES frame rates. Games should use this method to pause the game. 23 | } 24 | 25 | - (void)applicationDidEnterBackground:(UIApplication *)application 26 | { 27 | // Use this method to release shared resources, save user data, invalidate timers, and store enough application state information to restore your application to its current state in case it is terminated later. 28 | // If your application supports background execution, this method is called instead of applicationWillTerminate: when the user quits. 29 | } 30 | 31 | - (void)applicationWillEnterForeground:(UIApplication *)application 32 | { 33 | // Called as part of the transition from the background to the inactive state; here you can undo many of the changes made on entering the background. 34 | } 35 | 36 | - (void)applicationDidBecomeActive:(UIApplication *)application 37 | { 38 | // Restart any tasks that were paused (or not yet started) while the application was inactive. If the application was previously in the background, optionally refresh the user interface. 39 | } 40 | 41 | - (void)applicationWillTerminate:(UIApplication *)application 42 | { 43 | // Called when the application is about to terminate. Save data if appropriate. See also applicationDidEnterBackground:. 44 | } 45 | 46 | @end 47 | -------------------------------------------------------------------------------- /Example/ZLBridge/ZLBridge-Info.plist: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | CFBundleDevelopmentRegion 6 | en 7 | CFBundleDisplayName 8 | ${PRODUCT_NAME} 9 | CFBundleExecutable 10 | ${EXECUTABLE_NAME} 11 | CFBundleIdentifier 12 | $(PRODUCT_BUNDLE_IDENTIFIER) 13 | CFBundleInfoDictionaryVersion 14 | 6.0 15 | CFBundleName 16 | ${PRODUCT_NAME} 17 | CFBundlePackageType 18 | APPL 19 | CFBundleShortVersionString 20 | 1.0 21 | CFBundleSignature 22 | ???? 23 | CFBundleVersion 24 | 1.0 25 | LSRequiresIPhoneOS 26 | 27 | NSAppTransportSecurity 28 | 29 | NSAllowsArbitraryLoads 30 | 31 | 32 | UILaunchStoryboardName 33 | LaunchScreen 34 | UIMainStoryboardFile 35 | Main 36 | UIRequiredDeviceCapabilities 37 | 38 | armv7 39 | 40 | UISupportedInterfaceOrientations 41 | 42 | UIInterfaceOrientationPortrait 43 | UIInterfaceOrientationLandscapeLeft 44 | UIInterfaceOrientationLandscapeRight 45 | 46 | UISupportedInterfaceOrientations~ipad 47 | 48 | UIInterfaceOrientationPortrait 49 | UIInterfaceOrientationPortraitUpsideDown 50 | UIInterfaceOrientationLandscapeLeft 51 | UIInterfaceOrientationLandscapeRight 52 | 53 | 54 | 55 | -------------------------------------------------------------------------------- /Example/ZLBridge/ZLBridge-Prefix.pch: -------------------------------------------------------------------------------- 1 | // 2 | // Prefix header 3 | // 4 | // The contents of this file are implicitly included at the beginning of every source file. 5 | // 6 | 7 | #import 8 | 9 | #ifndef __IPHONE_5_0 10 | #warning "This project uses features only available in iOS SDK 5.0 and later." 11 | #endif 12 | 13 | #ifdef __OBJC__ 14 | @import UIKit; 15 | @import Foundation; 16 | #endif 17 | -------------------------------------------------------------------------------- /Example/ZLBridge/ZLTextVC.h: -------------------------------------------------------------------------------- 1 | // 2 | // ZLTextVC.h 3 | // ZLBridge_Example 4 | // 5 | // Created by 范鹏 on 2021/6/4. 6 | // Copyright © 2021 范鹏. All rights reserved. 7 | // 8 | 9 | #import 10 | 11 | NS_ASSUME_NONNULL_BEGIN 12 | 13 | @interface ZLTextVC : UIViewController 14 | 15 | @end 16 | 17 | NS_ASSUME_NONNULL_END 18 | -------------------------------------------------------------------------------- /Example/ZLBridge/ZLTextVC.m: -------------------------------------------------------------------------------- 1 | // 2 | // ZLTextVC.m 3 | // ZLBridge_Example 4 | // 5 | // Created by 范鹏 on 2021/6/4. 6 | // Copyright © 2021 范鹏. All rights reserved. 7 | // 8 | 9 | #import "ZLTextVC.h" 10 | #import 11 | #import 12 | #import "ZLWebview.h" 13 | @interface ZLTextVC () 14 | @property (strong, nonatomic) WKWebView *wkwebView; 15 | @property (strong,nonatomic)NSTimer *timer; 16 | @end 17 | @implementation ZLTextVC 18 | - (NSTimer *)timer { 19 | if (!_timer) { 20 | _timer = [NSTimer scheduledTimerWithTimeInterval:1 target:self selector:@selector(viewDidLoad) userInfo:nil repeats:YES]; 21 | } 22 | return _timer; 23 | } 24 | - (void)viewDidLoad 25 | { 26 | [super viewDidLoad]; 27 | self.wkwebView = [[ZLWebview alloc] initWithFrame:CGRectMake(0, 0, self.view.bounds.size.width, 300)]; 28 | [self.wkwebView initBridgeWithLocalJS:YES]; 29 | NSString *path = [NSBundle.mainBundle pathForResource:@"index.html" ofType:nil]; 30 | NSURL *url = [NSURL fileURLWithPath:path]; 31 | [self.wkwebView loadRequest:[NSURLRequest requestWithURL:url]]; 32 | [self.view addSubview:self.wkwebView]; 33 | __weak typeof(self) weakSelf = self; 34 | 35 | [self.wkwebView registHandler:@"test" completionHandler:^(id _Nullable obj, JSCallbackHandler _Nullable callback) { 36 | callback(@"js异步调用:这是原生返回的结果1000!",NO); 37 | }]; 38 | [self.wkwebView registHandler:@"upload" completionHandler:^(id _Nullable obj, JSCallbackHandler _Nullable callback) { 39 | [weakSelf uploadCompletionHandler:callback]; 40 | }]; 41 | [self.wkwebView callHandler:@"jsMethod" arguments:@[@"这是原生调用js传的值"] completionHandler:^(id _Nullable obj, NSString * _Nullable error) { 42 | NSString *msg = [obj isKindOfClass:NSString.class] ? obj : [ZLUtils objToJsonString:obj]; 43 | }]; 44 | 45 | } 46 | #pragma mark - 原生主动调用js 47 | - (IBAction)calljs:(UIButton*)sender { 48 | [self.wkwebView callHandler:@"jsMethod" arguments:@[@"这是原生主动调用js原生传给js的值,js原封不动把值返回"] completionHandler:^(id _Nullable obj, NSString * _Nullable error) { 49 | NSLog(@"%@",obj); 50 | NSString *msg = [obj isKindOfClass:NSString.class] ? obj : [ZLUtils objToJsonString:obj]; 51 | [sender setTitle: msg forState:UIControlStateNormal]; 52 | }]; 53 | } 54 | - (void)uploadCompletionHandler:(JSCallbackHandler _Nullable)completionHandler { 55 | __block int i = 0; 56 | if (self.timer) { 57 | [self.timer invalidate]; 58 | self.timer = nil; 59 | } 60 | self.timer = [NSTimer scheduledTimerWithTimeInterval:1 repeats:YES block:^(NSTimer * _Nonnull timer) { 61 | i += 1; 62 | BOOL end = i == 10; 63 | NSString *str = end ? @"上传完成" : [NSString stringWithFormat:@"%d%%",i*10]; 64 | completionHandler(str,end); 65 | }]; 66 | } 67 | 68 | @end 69 | -------------------------------------------------------------------------------- /Example/ZLBridge/ZLTextVC.xib: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15 | 16 | 17 | 18 | 19 | 20 | 21 | 22 | 23 | -------------------------------------------------------------------------------- /Example/ZLBridge/ZLViewController.h: -------------------------------------------------------------------------------- 1 | // 2 | // ZLViewController.h 3 | // ZLBridge 4 | // 5 | // Created by 范鹏 on 06/04/2021. 6 | // Copyright (c) 2021 范鹏. All rights reserved. 7 | // 8 | 9 | @import UIKit; 10 | 11 | @interface ZLViewController : UIViewController 12 | 13 | @end 14 | -------------------------------------------------------------------------------- /Example/ZLBridge/ZLViewController.m: -------------------------------------------------------------------------------- 1 | // 2 | // ZLViewController.m 3 | // ZLBridge 4 | // 5 | // Created by 范鹏 on 06/04/2021. 6 | // Copyright (c) 2021 范鹏. All rights reserved. 7 | // 8 | #import "ZLViewController.h" 9 | #import 10 | #import 11 | #import "ZLTextVC.h" 12 | @interface ZLViewController () 13 | @property (strong, nonatomic) WKWebView *wkwebView; 14 | @property (strong,nonatomic)NSTimer *timer; 15 | @end 16 | @implementation ZLViewController 17 | - (NSTimer *)timer { 18 | if (!_timer) { 19 | _timer = [NSTimer scheduledTimerWithTimeInterval:1 target:self selector:@selector(viewDidLoad) userInfo:nil repeats:YES]; 20 | } 21 | return _timer; 22 | } 23 | - (void)viewDidLoad 24 | { 25 | [super viewDidLoad]; 26 | self.wkwebView = [[WKWebView alloc] initWithFrame:CGRectMake(0, 0, self.view.bounds.size.width, 300)]; 27 | [self.wkwebView initBridgeWithLocalJS:YES]; 28 | self.wkwebView.UIDelegate = self; 29 | NSString *path = [NSBundle.mainBundle pathForResource:@"index.html" ofType:nil]; 30 | NSURL *url = [NSURL fileURLWithPath:path]; 31 | // url = [NSURL URLWithString:@"http://localhost:3000/"]; 32 | [self.wkwebView loadRequest:[NSURLRequest requestWithURL:url]]; 33 | [self.view addSubview:self.wkwebView]; 34 | [self.wkwebView registHandler:@"test" completionHandler:^(id _Nullable obj, JSCallbackHandler _Nullable callback) { 35 | callback(obj,YES); 36 | }]; 37 | [self.wkwebView registUndefinedHandler:^(NSString * _Nullable name, id _Nullable obj, JSCallbackHandler _Nullable callback) { 38 | NSLog(@"registUndefinedHandlerCompletionHandler:%@",name); 39 | }]; 40 | [self.wkwebView registHandler:@"upload" completionHandler:^(id _Nullable obj, JSCallbackHandler _Nullable callback) { 41 | [self uploadCompletionHandler:callback]; 42 | }]; 43 | } 44 | - (void)webView:(WKWebView *)webView runJavaScriptAlertPanelWithMessage:(NSString *)message initiatedByFrame:(WKFrameInfo *)frame completionHandler:(void (^)(void))completionHandler{ 45 | completionHandler(); 46 | } 47 | #pragma mark - 原生主动调用js 48 | - (IBAction)calljs:(UIButton*)sender { 49 | [self.wkwebView callHandler:@"jsMethod" arguments:@[@"这是原生调用js传的值"] completionHandler:^(id _Nullable obj, NSString * _Nullable error) { 50 | NSString *msg; 51 | if (error) { 52 | msg = error; 53 | }else { 54 | msg = [obj isKindOfClass:NSString.class] ? obj : [ZLUtils objToJsonString:obj]; 55 | } 56 | [sender setTitle: msg forState:UIControlStateNormal]; 57 | }]; 58 | } 59 | - (void)uploadCompletionHandler:(JSCallbackHandler _Nullable)completionHandler { 60 | __block int i = 0; 61 | if (self.timer) { 62 | [self.timer invalidate]; 63 | self.timer = nil; 64 | } 65 | self.timer = [NSTimer scheduledTimerWithTimeInterval:1 repeats:YES block:^(NSTimer * _Nonnull timer) { 66 | i += 1; 67 | BOOL end = i == 10; 68 | NSString *str = end ? @"上传完成" : [NSString stringWithFormat:@"%d%%",i*10]; 69 | completionHandler(str,end); 70 | }]; 71 | } 72 | - (IBAction)refresh:(id)sender { 73 | [self.wkwebView reload]; 74 | } 75 | - (IBAction)jsAction1:(id)sender { 76 | [self.wkwebView callHandler:@"jsMethod" completionHandler:^(id _Nullable obj, NSString * _Nullable error) { 77 | NSString *msg = error ? error : @"成功调用JS事件1"; 78 | [sender setTitle:msg forState:UIControlStateNormal]; 79 | }]; 80 | } 81 | - (IBAction)jsAction2:(UIButton*)sender { 82 | [self.wkwebView callHandler:@"jsMethodWithCallback" arguments:nil completionHandler:^(id _Nullable obj, NSString * _Nullable error) { 83 | NSString *msg = error ? error : @"成功调用JS事件2"; 84 | [sender setTitle:msg forState:UIControlStateNormal]; 85 | }]; 86 | } 87 | - (void)dealloc 88 | { 89 | [self.wkwebView destroyBridge]; 90 | } 91 | @end 92 | -------------------------------------------------------------------------------- /Example/ZLBridge/ZLWebview.h: -------------------------------------------------------------------------------- 1 | // 2 | // ZLWebview.h 3 | // ZLBridge_Example 4 | // 5 | // Created by 范鹏 on 2021/6/4. 6 | // Copyright © 2021 范鹏. All rights reserved. 7 | // 8 | 9 | #import 10 | 11 | NS_ASSUME_NONNULL_BEGIN 12 | 13 | @interface ZLWebview : WKWebView 14 | 15 | @end 16 | 17 | NS_ASSUME_NONNULL_END 18 | -------------------------------------------------------------------------------- /Example/ZLBridge/ZLWebview.m: -------------------------------------------------------------------------------- 1 | // 2 | // ZLWebview.m 3 | // ZLBridge_Example 4 | // 5 | // Created by 范鹏 on 2021/6/4. 6 | // Copyright © 2021 范鹏. All rights reserved. 7 | // 8 | 9 | #import "ZLWebview.h" 10 | #import 11 | #import 12 | @implementation ZLWebview 13 | 14 | /* 15 | // Only override drawRect: if you perform custom drawing. 16 | // An empty implementation adversely affects performance during animation. 17 | - (void)drawRect:(CGRect)rect { 18 | // Drawing code 19 | } 20 | */ 21 | - (void)dealloc 22 | { 23 | NSLog(@"webview已销毁"); 24 | } 25 | 26 | @end 27 | -------------------------------------------------------------------------------- /Example/ZLBridge/en.lproj/InfoPlist.strings: -------------------------------------------------------------------------------- 1 | /* Localized versions of Info.plist keys */ 2 | 3 | -------------------------------------------------------------------------------- /Example/ZLBridge/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 12 | 51 | Document 52 | 53 | 54 | 55 | 56 | 57 | 58 |
59 |
60 | 61 | 62 | 63 | 64 | -------------------------------------------------------------------------------- /Example/ZLBridge/main.m: -------------------------------------------------------------------------------- 1 | // 2 | // main.m 3 | // ZLBridge 4 | // 5 | // Created by 范鹏 on 06/04/2021. 6 | // Copyright (c) 2021 范鹏. All rights reserved. 7 | // 8 | 9 | @import UIKit; 10 | #import "ZLAppDelegate.h" 11 | 12 | int main(int argc, char * argv[]) 13 | { 14 | @autoreleasepool { 15 | return UIApplicationMain(argc, argv, nil, NSStringFromClass([ZLAppDelegate class])); 16 | } 17 | } 18 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | Copyright (c) 2021 范鹏 <2551412939@qq.com> 2 | 3 | Permission is hereby granted, free of charge, to any person obtaining a copy 4 | of this software and associated documentation files (the "Software"), to deal 5 | in the Software without restriction, including without limitation the rights 6 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 7 | copies of the Software, and to permit persons to whom the Software is 8 | furnished to do so, subject to the following conditions: 9 | 10 | The above copyright notice and this permission notice shall be included in 11 | all copies or substantial portions of the Software. 12 | 13 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 14 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 15 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 16 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 17 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 18 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN 19 | THE SOFTWARE. 20 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # ZLBridge 2 | 3 | [![CI Status](https://img.shields.io/travis/范鹏/ZLBridge.svg?style=flat)](https://travis-ci.org/范鹏/ZLBridge) 4 | [![Version](https://img.shields.io/cocoapods/v/ZLBridge.svg?style=flat)](https://cocoapods.org/pods/ZLBridge) 5 | [![License](https://img.shields.io/cocoapods/l/ZLBridge.svg?style=flat)](https://cocoapods.org/pods/ZLBridge) 6 | [![Platform](https://img.shields.io/cocoapods/p/ZLBridge.svg?style=flat)](https://cocoapods.org/pods/ZLBridge) 7 | 8 | ## 说明 9 | ZLBridge是为iOS的WKWebview和JS数据交互时提供更简单方便的小工具组件,可配合H5端使用ZLBridge-JS库来数据交互,也可选择原生本地注入JS代码,H5无需任何集成操作。目前已支持的平台有 10 |
[ZLBridge-iOS](https://github.com/FPJack/ZLBridge-iOS) 11 |
[ZLBridge-Android](https://github.com/FPJack/ZLBridge-Android) 12 |
[ZLBridge-JS](https://github.com/FPJack/ZLBridge-JS) 13 |
[ZLBridge-flutter](https://github.com/FPJack/ZLBridge-flutter) 14 |
[ZLBridge-RN](https://github.com/FPJack/ZLBridge-RN) 15 | 16 | ## 安装 17 | ```ruby 18 | 19 | pod 'ZLBridge' //如果安装失败,pod update更新一下 20 | ``` 21 | ## H5端window.zlbridge初始化 22 | ```objective-c 23 | #import 24 | 25 | //YES:原生注入本地js脚本初始化zlbridge,NO:由H5初始化zlbridge 26 | [self.wkwebView initBridgeWithLocalJS:YES]; 27 | ``` 28 | H5初始化zlbridge 29 | ```JavaScript 30 | //导入一次后也可以通过window.zlbridge拿zlbridge对象 31 | var zlbridge = require('zlbridge') 32 | ``` 33 | ## 原生与JS交互 34 | 35 | 36 | ### JS调用原生test事件 37 | 38 | #### 无参数 39 | ```JavaScript 40 | window.zlbridge.call('test',(arg) => { 41 | 42 | }); 43 | ``` 44 | #### 有参数参数 45 | ```JavaScript 46 | window.zlbridge.call('test',{key:"value"},(arg) => { 47 | 48 | }); 49 | ``` 50 | #### 原生注册test事件 51 | ```objective-c 52 | [self.wkwebView registHandler:@"test" completionHandler:^(id _Nullable obj, JSCallbackHandler _Nullable callback) { 53 | //YES代表JS只能监听一次回调结果,NO可以连续监听 54 | callback(@"js异步调用:这是原生返回的结果1000!",YES); 55 | }]; 56 | ``` 57 | 58 | 59 | ### 原生调用js 60 | 61 | #### 原生调用JS的jsMethod事件 62 | ```objective-c 63 | [self.wkwebView callHandler:@"jsMethod" arguments:@[@"这是原生调用js传的值"] completionHandler:^(id _Nullable obj, NSError * _Nullable error) { 64 | }]; 65 | ``` 66 | 67 | #### js注册jsMethod事件 68 | ```JavaScript 69 | window.zlbridge.register("jsMethod",(arg) => { 70 | return arg; 71 | }); 72 | ``` 73 | 或者 74 | ```JavaScript 75 | window.zlbridge.registerWithCallback("jsMethod",(arg,callback) => { 76 | //ture代表原生只能监听一次回调结果,false可以连续监听,默认传为true 77 | callback(arg,true); 78 | }); 79 | ``` 80 | 81 | ## 通过本地注入JS脚本的,H5可以监听ZLBridge初始化完成 82 | ```JavaScript 83 | document.addEventListener('ZLBridgeInitReady', function() { 84 | consloe.log('ZLBridge初始化完成'); 85 | },false); 86 | ``` 87 | 88 | ## 移除ZLBridge 89 | ```objective-c 90 | [self.wkwebView destroyBridge]; 91 | ``` 92 | ## !!! iOS传给JS的值对象有NSNumber,NSString,NSDictionary,NSArray,NSSet 93 | ## Author 94 | 95 | 范鹏, 2551412939@qq.com 96 | 97 | 98 | 99 | ## License 100 | 101 | ZLBridge is available under the MIT license. See the LICENSE file for more info. 102 | -------------------------------------------------------------------------------- /ZLBridge.podspec: -------------------------------------------------------------------------------- 1 | # 2 | # Be sure to run `pod lib lint ZLBridge.podspec' to ensure this is a 3 | # valid spec before submitting. 4 | # 5 | # Any lines starting with a # are optional, but their use is encouraged 6 | # To learn more about a Podspec see https://guides.cocoapods.org/syntax/podspec.html 7 | # 8 | 9 | Pod::Spec.new do |s| 10 | s.name = 'ZLBridge' 11 | s.version = '1.0.0' 12 | s.summary = 'A short description of ZLBridge.' 13 | 14 | # This description is used to generate tags and improve search results. 15 | # * Think: What does it do? Why did you write it? What is the focus? 16 | # * Try to keep it short, snappy and to the point. 17 | # * Write the description between the DESC delimiters below. 18 | # * Finally, don't worry about the indent, CocoaPods strips it! 19 | 20 | s.description = <<-DESC 21 | TODO: Add long description of the pod here. 22 | DESC 23 | 24 | s.homepage = 'https://github.com/FPJack/ZLBridge.git' 25 | # s.screenshots = 'www.example.com/screenshots_1', 'www.example.com/screenshots_2' 26 | s.license = { :type => 'MIT', :file => 'LICENSE' } 27 | s.author = { '范鹏' => '2551412939@qq.com' } 28 | s.source = { :git => 'https://github.com/FPJack/ZLBridge.git', :tag => s.version.to_s } 29 | # s.social_media_url = 'https://twitter.com/' 30 | 31 | s.ios.deployment_target = '9.0' 32 | 33 | s.source_files = 'ZLBridge/Classes/**/*' 34 | # s.resource = ['ZLBridge/Assets/*'] 35 | 36 | s.resource_bundles = { 37 | 'ZLBridge' => ['ZLBridge/Assets/*'] 38 | } 39 | 40 | # s.public_header_files = 'Pod/Classes/**/*.h' 41 | # s.frameworks = 'UIKit', 'MapKit' 42 | # s.dependency 'AFNetworking', '~> 2.3' 43 | end 44 | -------------------------------------------------------------------------------- /ZLBridge/Assets/.gitkeep: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/FPJack/ZLBridge-iOS/85cdbb62eeae25b02f2681d07456c0c501d1913b/ZLBridge/Assets/.gitkeep -------------------------------------------------------------------------------- /ZLBridge/Assets/ZLBridge.js: -------------------------------------------------------------------------------- 1 | (function () { 2 | if (window.zlbridge) return ; 3 | var zlbridge = { 4 | call: function(method,arg,func){ 5 | if (typeof method != 'string') return; 6 | if (typeof arg == 'function') { 7 | func = arg; 8 | arg = null; 9 | } 10 | var args = {}; 11 | args['body'] = arg; 12 | args['name'] = method; 13 | args['end'] = true; 14 | args['callID'] = ''; 15 | if (typeof func == 'function') { 16 | var _callHandlerID = setTimeout(function(){}); 17 | _callHandlerID = '_methodid_' + _callHandlerID + new Date().getTime(); 18 | args['jsMethodId'] = _callHandlerID ; 19 | window.zlbridge[_callHandlerID] = func; 20 | } 21 | window.zlbridge._callNative(args); 22 | }, 23 | _callNative: function(arg) { 24 | var json = JSON.stringify(arg); 25 | if(window.ZLBridge && window.ZLBridge.postMessage){ 26 | window.ZLBridge.postMessage(json); 27 | }else if (window.webkit && window.webkit.messageHandlers && window.webkit.messageHandlers.ZLBridge){ 28 | window.webkit.messageHandlers.ZLBridge.postMessage(json); 29 | }else if(window.ReactNativeWebView && window.ReactNativeWebView.postMessage) { 30 | window.ReactNativeWebView.postMessage(json); 31 | } 32 | }, 33 | register: function(method,func){ 34 | if (typeof func == 'function' && typeof method == 'string') { 35 | window.zlbridge['_register_' + method] = func; 36 | } 37 | }, 38 | removeRegisted: function(method) { 39 | if (typeof method != 'string') return; 40 | delete window.zlbridge['_register_' + method]; 41 | delete window.zlbridge['_register_callback' + method]; 42 | }, 43 | registerWithCallback: function(method,func){ 44 | if (typeof func == 'function' && typeof method == 'string') { 45 | window.zlbridge['_register_callback' + method] = func; 46 | } 47 | }, 48 | _nativeCall: function(method,arg) { 49 | var obj = JSON.parse(arg); 50 | var result = obj['result']; 51 | var callID = obj['callID']; 52 | setTimeout(() => { 53 | try { 54 | var func = window.zlbridge['_register_' + method]; 55 | if (typeof func == 'function') { 56 | var args = {}; 57 | args['end'] = true; 58 | if (callID) args['callID'] = callID; 59 | args['body'] = func(result); 60 | return window.zlbridge._callNative(args); 61 | } 62 | func = window.zlbridge['_register_callback' + method]; 63 | var callback = function (params,end) { 64 | var args = {}; 65 | if (callID) args['callID'] = callID; 66 | args['body'] = params; 67 | args['end'] = (typeof end == 'boolean')?end:true; 68 | window.zlbridge._callNative(args); 69 | }; 70 | func(result,callback); 71 | } catch (error) { 72 | window.zlbridge._callNative({error:error.message,callID:callID,end:true}); 73 | } 74 | }); 75 | }, 76 | _nativeCallback: function(methodid,arg){ 77 | setTimeout(() => { 78 | var func = window.zlbridge[methodid]; 79 | if (typeof func != 'function') return; 80 | arg = JSON.parse(arg); 81 | func(arg['result']); 82 | if (arg.end==1) delete window.zlbridge[methodid]; 83 | }); 84 | }, 85 | _hasNativeMethod: function(method) { 86 | var func = window.zlbridge['_register_' + method]; 87 | if (typeof func != 'function')func = window.zlbridge['_register_callback' + method]; 88 | return (func!=null||func!=undefined); 89 | } 90 | }; 91 | window.zlbridge = zlbridge; 92 | var doc = document; 93 | var event = doc.createEvent('Events'); 94 | event.initEvent('ZLBridgeInitReady'); 95 | doc.dispatchEvent(event); 96 | })(); 97 | -------------------------------------------------------------------------------- /ZLBridge/Classes/.gitkeep: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/FPJack/ZLBridge-iOS/85cdbb62eeae25b02f2681d07456c0c501d1913b/ZLBridge/Classes/.gitkeep -------------------------------------------------------------------------------- /ZLBridge/Classes/WKWebView+ZLBridge.h: -------------------------------------------------------------------------------- 1 | // 2 | // WKWebView+ZLWebView.h 3 | // ZLBridge 4 | // 5 | // Created by 范鹏 on 2021/6/4. 6 | // 7 | 8 | #import 9 | typedef void (^JSCompletionHandler)(id _Nullable obj, NSString * _Nullable error); 10 | typedef void (^JSCallbackHandler)(id _Nullable obj, BOOL end); 11 | typedef void (^JSRegistHandler)(id _Nullable obj,JSCallbackHandler _Nullable callback); 12 | typedef void (^JSRegistUndefinedHandler)(NSString * _Nullable name,id _Nullable obj,JSCallbackHandler _Nullable callback); 13 | 14 | NS_ASSUME_NONNULL_BEGIN 15 | @interface ZLUtils : NSObject 16 | + (NSString * _Nullable)objToJsonString:(id _Nonnull)dict; 17 | + (id _Nullable)jsonStringToObject:(NSString * _Nonnull)jsonString; 18 | @end 19 | NS_ASSUME_NONNULL_END 20 | 21 | NS_ASSUME_NONNULL_BEGIN 22 | @interface ZLMsgBody : NSObject 23 | @property (nonatomic,copy)NSString *name; 24 | @property (nonatomic,copy)NSString *jsMethodId; 25 | @property (nonatomic,strong)id body; 26 | @property (nonatomic,copy)NSString *callID; 27 | @property (nonatomic,copy)NSString *end; 28 | @property (nonatomic,copy)NSString *error; 29 | + (instancetype)initMsgBodyWithDic:(NSDictionary*)dic; 30 | @end 31 | NS_ASSUME_NONNULL_END 32 | 33 | NS_ASSUME_NONNULL_BEGIN 34 | @interface ZLBridge : NSObject 35 | @property (nonatomic,copy)void(^msgCallback)(ZLMsgBody *message); 36 | @end 37 | NS_ASSUME_NONNULL_END 38 | 39 | NS_ASSUME_NONNULL_BEGIN 40 | @interface WKWebView (ZLBridge) 41 | //初始化bridge的时候是否本地注入js 42 | - (void)initBridgeWithLocalJS:(BOOL )localJs; 43 | //需要手动移除bridge,否则会内存泄漏 44 | - (void)destroyBridge; 45 | //注册监听js调用事件 46 | -(void) registHandler:(NSString * _Nonnull) methodName completionHandler:(JSRegistHandler _Nonnull) registHandler; 47 | //移除相应的js监听事件 48 | - (void)removeRegistedHandlerWithMethodName:(NSString *)methodName; 49 | //移除所有监听js事件 50 | - (void)removeAllRegistedHandler; 51 | //注册未定义的原生事件回调block 52 | -(void) registUndefinedHandler:(JSRegistUndefinedHandler _Nonnull) registHandler; 53 | //原生调用js 54 | -(void) callHandler:(NSString * _Nonnull) methodName completionHandler:(JSCompletionHandler _Nonnull)completionHandler; 55 | //原生调用js有参数 56 | -(void) callHandler:(NSString * _Nonnull) methodName arguments:(NSArray * _Nullable) args completionHandler:(JSCompletionHandler _Nonnull)completionHandler; 57 | //js是否注册了原生调用的方法 58 | - (void)hasNativeMethod:(NSString * _Nonnull)methodName callback:(void(^ _Nullable)(BOOL exist))callback; 59 | @end 60 | NS_ASSUME_NONNULL_END 61 | -------------------------------------------------------------------------------- /ZLBridge/Classes/WKWebView+ZLBridge.m: -------------------------------------------------------------------------------- 1 | // 2 | // WKWebView+ZLWebView.m 3 | // ZLBridge 4 | // 5 | // Created by 范鹏 on 2021/6/4. 6 | // 7 | 8 | #import "WKWebView+ZLBridge.h" 9 | #import 10 | #define kUndefinedHandler @"kUndefinedHandler" 11 | @implementation ZLUtils 12 | + (NSString *)objToJsonString:(id)dict 13 | { 14 | NSString *jsonString = nil; 15 | NSError *error; 16 | if (![NSJSONSerialization isValidJSONObject:dict]) return @"{}"; 17 | NSData *jsonData = [NSJSONSerialization dataWithJSONObject:dict options:0 error:&error]; 18 | if (!jsonData) return @"{}"; 19 | jsonString = [[NSString alloc] initWithData:jsonData encoding:NSUTF8StringEncoding]; 20 | return jsonString; 21 | } 22 | + (id )jsonStringToObject:(NSString *)jsonString 23 | { 24 | if (jsonString == nil) return nil; 25 | NSData *jsonData = [jsonString dataUsingEncoding:NSUTF8StringEncoding]; 26 | NSError *err; 27 | NSDictionary *dic = [NSJSONSerialization JSONObjectWithData:jsonData 28 | options:NSJSONReadingMutableContainers 29 | error:&err]; 30 | if(err) return nil; 31 | return dic; 32 | } 33 | @end 34 | ////// 35 | ////// 36 | @implementation ZLMsgBody 37 | + (instancetype)initMsgBodyWithDic:(NSDictionary*)dic{ 38 | ZLMsgBody *obj = [[ZLMsgBody alloc] init]; 39 | obj.name = dic[@"name"]; 40 | obj.body = dic[@"body"]; 41 | obj.callID = dic[@"callID"]; 42 | obj.error = dic[@"error"]; 43 | obj.end = [NSString stringWithFormat:@"%@",dic[@"end"]]; 44 | obj.jsMethodId = dic[@"jsMethodId"]; 45 | return obj; 46 | } 47 | @end 48 | ///////////////////// 49 | //////////////////// 50 | @implementation ZLBridge 51 | - (void)userContentController:(WKUserContentController *)userContentController didReceiveScriptMessage:(WKScriptMessage *)message{ 52 | NSDictionary *body = [ZLUtils jsonStringToObject:message.body]; 53 | ZLMsgBody *obj = [ZLMsgBody initMsgBodyWithDic:body]; 54 | if (self.msgCallback) self.msgCallback(obj); 55 | } 56 | @end 57 | ///////////////////// 58 | //////////////////// 59 | @implementation WKWebView (ZLBridge) 60 | static const char JSCompletionHandlersKey = '\0'; 61 | - (void)setRegistHanders:(NSMutableDictionary *)registHanders { 62 | objc_setAssociatedObject(self, &JSCompletionHandlersKey, 63 | registHanders,OBJC_ASSOCIATION_RETAIN_NONATOMIC); 64 | } 65 | - (NSMutableDictionary *)registHanders { 66 | NSMutableDictionary *dic = objc_getAssociatedObject(self, &JSCompletionHandlersKey); 67 | if (dic == nil) { 68 | dic = [NSMutableDictionary dictionary]; 69 | self.registHanders = dic; 70 | } 71 | return dic; 72 | } 73 | 74 | static const char JSCallHandlersKey = '\0'; 75 | - (void)setCallHanders:(NSMutableDictionary *)callHanders { 76 | objc_setAssociatedObject(self, &JSCallHandlersKey, 77 | callHanders,OBJC_ASSOCIATION_RETAIN_NONATOMIC); 78 | } 79 | - (NSMutableDictionary *)callHanders { 80 | NSMutableDictionary *dic = objc_getAssociatedObject(self, &JSCallHandlersKey); 81 | if (dic == nil) { 82 | dic = [NSMutableDictionary dictionary]; 83 | self.callHanders = dic; 84 | } 85 | return dic; 86 | } 87 | - (void)initBridgeWithLocalJS:(BOOL )localJs{ 88 | if (localJs) { 89 | NSBundle *selfBundle = [NSBundle bundleForClass:ZLBridge.class]; 90 | NSString *path = [selfBundle pathForResource:@"ZLBridge.bundle/ZLBridge.js" ofType:nil]; 91 | NSData *data=[NSData dataWithContentsOfFile:path]; 92 | NSString *js = [[NSString alloc]initWithData:data encoding:NSUTF8StringEncoding]; 93 | WKUserScript *jsScript = [[WKUserScript alloc] initWithSource:js 94 | injectionTime:WKUserScriptInjectionTimeAtDocumentStart 95 | forMainFrameOnly:YES]; 96 | [self.configuration.userContentController addUserScript:jsScript]; 97 | } 98 | 99 | ZLBridge *bridge = [[ZLBridge alloc] init]; 100 | __weak typeof(self) weakSelf = self; 101 | bridge.msgCallback = ^(ZLMsgBody * _Nonnull message) { 102 | NSString *name = message.name; 103 | NSString *callID = message.callID; 104 | NSString *end = message.end; 105 | NSString *error = message.error; 106 | id body = message.body; 107 | if (callID && callID.length > 0) { 108 | JSCompletionHandler callHandler = weakSelf.callHanders[callID]; 109 | if (callHandler) { 110 | dispatch_async(dispatch_get_main_queue(), ^{ 111 | callHandler(body,error); 112 | if ([end isKindOfClass:NSString.class] && [end isEqualToString:@"1"]) { 113 | [weakSelf.callHanders removeObjectForKey:callID]; 114 | } 115 | }); 116 | } 117 | return; 118 | } 119 | NSString *jsMethodId = message.jsMethodId; 120 | JSRegistHandler registHandler = weakSelf.registHanders[name]; 121 | JSCallbackHandler callBack = ^(id result,BOOL end){ 122 | NSMutableDictionary *mDic = NSMutableDictionary.dictionary; 123 | mDic[@"end"] = end?@1:@0; 124 | mDic[@"result"] = result?result:@""; 125 | NSString *js = [NSString stringWithFormat:@"window.zlbridge._nativeCallback('%@','%@');",jsMethodId,[ZLUtils objToJsonString:mDic]]; 126 | [weakSelf evaluateJavaScript:js completionHandler:nil]; 127 | }; 128 | dispatch_async(dispatch_get_main_queue(), ^{ 129 | if (registHandler) { 130 | registHandler(body,callBack); 131 | }else { 132 | JSRegistUndefinedHandler registUndefinedHandler= weakSelf.registHanders[kUndefinedHandler]; 133 | if (registUndefinedHandler) registUndefinedHandler(name,body,callBack); 134 | } 135 | }); 136 | }; 137 | [self.configuration.userContentController addScriptMessageHandler:bridge name:@"ZLBridge"]; 138 | } 139 | - (void)destroyBridge{ 140 | [self.registHanders removeAllObjects]; 141 | [self.callHanders removeAllObjects]; 142 | [self.configuration.userContentController removeScriptMessageHandlerForName:@"ZLBridge"]; 143 | } 144 | -(void) registHandler:(NSString * _Nonnull) methodName completionHandler:(JSRegistHandler _Nonnull) registHandler{ 145 | if (![methodName isKindOfClass:NSString.class] || !registHandler) return; 146 | self.registHanders[methodName] = registHandler; 147 | } 148 | - (void)removeRegistedHandlerWithMethodName:(NSString *)methodName{ 149 | if (!methodName) return; 150 | [self.registHanders removeObjectForKey:methodName]; 151 | } 152 | - (void)removeAllRegistedHandler{ 153 | [self.registHanders removeAllObjects]; 154 | } 155 | -(void) registUndefinedHandler:(JSRegistUndefinedHandler _Nonnull) registHandler{ 156 | if (registHandler) self.registHanders[kUndefinedHandler] = registHandler; 157 | } 158 | -(void)callHandler:(NSString * _Nonnull) methodName completionHandler:(JSCompletionHandler _Nonnull)completionHandler{ 159 | [self callHandler:methodName arguments:nil completionHandler:completionHandler]; 160 | } 161 | -(void)callHandler:(NSString * _Nonnull) methodName arguments:(NSArray * _Nullable) args completionHandler:(JSCompletionHandler _Nonnull)completionHandler{ 162 | args = args == nil ? @[] : args; 163 | NSMutableDictionary *dic = [NSMutableDictionary dictionary]; 164 | if (args) dic[@"result"] = args; 165 | dispatch_async(dispatch_get_main_queue(), ^{ 166 | NSString *ID; 167 | if (completionHandler) { 168 | ID = [NSString stringWithFormat:@"%f",[[NSDate date] timeIntervalSince1970]]; 169 | dic[@"callID"] = ID; 170 | self.callHanders[ID] = completionHandler; 171 | } 172 | NSString *js = [NSString stringWithFormat:@"window.zlbridge._nativeCall('%@','%@');",methodName,[ZLUtils objToJsonString:dic]]; 173 | __weak typeof(self) weakSelf = self; 174 | [self evaluateJavaScript:js completionHandler:^(id _Nullable obj, NSError * _Nullable error) { 175 | if (error && completionHandler) { 176 | completionHandler(nil,error.description); 177 | if (ID) [weakSelf.callHanders removeObjectForKey:ID]; 178 | } 179 | }]; 180 | }); 181 | } 182 | - (void)hasNativeMethod:(NSString * _Nonnull)methodName callback:(void(^ _Nullable)(BOOL exist))callback;{ 183 | if (!callback) return; 184 | if (methodName == nil || methodName.length == 0 ) { 185 | callback(NO); 186 | return; 187 | } 188 | NSString *js = [NSString stringWithFormat:@"window.zlbridge._hasNativeMethod('%@');",methodName]; 189 | [self evaluateJavaScript:js completionHandler:^(NSNumber * _Nullable obj, NSError * _Nullable error) { 190 | if (error) { 191 | callback(NO); 192 | }else { 193 | callback(obj.boolValue); 194 | } 195 | }]; 196 | } 197 | @end 198 | -------------------------------------------------------------------------------- /_Pods.xcodeproj: -------------------------------------------------------------------------------- 1 | Example/Pods/Pods.xcodeproj --------------------------------------------------------------------------------