This article mainly introduces the iOS loading PDF method.

Load using webView

Using webView loading is the simplest way, this is the biggest advantage, easy to use, and can be added to the network file, the disadvantage is that the effect experience is not very good, a single up and down sliding effect.

Note:

The UIWebview component has been disabled by Apple this year, so webView cannot be submitted to The App Store. What is used in this article is based on WKWebView.

So let’s first look at the webView loading method,

– (nullable WKNavigation *)loadFileURL:(NSURL *)URL allowingReadAccessToURL:(NSURL *)readAccessURL

The first argument to this method is, of course, the path to the file we want to load. Note apple’s explanation that this is a file URL, and we should use the result of this method as an argument.

    NSURL *fileUrl = [NSURL fileURLWithPath:@"path"];
Copy the code

The second parameter, readAccessURL, is used to pass in the path of the resource referenced by the page. Normally, it is the parent directory of the previous File URL, and the remarks are also detailed. If this parameter is passed in a single File, only that File can be loaded by WebKit.

If the PDF is garbled, use the following method to load it:

   [self.wkWebView loadData:data MIMEType:@"application/pdf" characterEncodingName:@"GBK" baseURL:fileUrl];	
Copy the code

Using QLPreviewController

Using QLPreviewController requires importing the QuickLook framework

#import <QuickLook/QuickLook.h>		
Copy the code

Follow QLPreviewControllerDelegate and QLPreviewControllerDataSource,

The usage is as follows:

@property (nonatomic, strong) QLPreviewController *previewController;
Copy the code
- (QLPreviewController *)previewController {
    if(! _previewController) { _previewController = [[QLPreviewController alloc] init]; _previewController.delegate =self;
        _previewController.dataSource = self;
    }
    return _previewController;
}	
Copy the code

Implementation protocol:

/ / / returns the number of files - (NSInteger) numberOfPreviewItemsInPreviewController: (QLPreviewController *) controller {return 1; } - (id<QLPreviewItem>)previewController:(QLPreviewController *)controller previewItemAtIndex:(NSInteger)index { NSString *pdfPath = [[NSBundle mainBundle] pathForResource:@"test" ofType:@"pdf"]; return [NSURL fileURLWithPath:pdfPath]; }Copy the code

Another way is to use UIDocumentInteractionController preview, effect and QLPreviewController, but does not support multiple file preview.

Also need to follow when using UIDocumentInteractionControllerDelegate.

- (UIViewController *)documentInteractionControllerViewControllerForPreview:(UIDocumentInteractionController *)controller {
    return self;
}
- (UIView*)documentInteractionControllerViewForPreview:(UIDocumentInteractionController*)controller {
    return self.view;
}
- (CGRect)documentInteractionControllerRectForPreview:(UIDocumentInteractionController*)controller {
    return self.view.frame;
}
Copy the code

UIDocumentInteractionController inherits from NSObject, rather than the UIViewController, show the class used to own presentPreviewAnimated method.

The advantage of these two loading methods is,

  1. Easy integration, and its own system sharing function

  2. QLPreviewController supports multi-file preview

The disadvantage is that the experience is poor, and webView similar to the slide up and down preview

CGContexDrawPDFPage

If we need a good experience, like the animation of flipping a book, then we need to use CGContexDrawPDFPage for our own drawing.

  1. Obtaining PDF files

    / / / access path CFStringRef path = CFStringCreateWithCString (NULL, filePath. UTF8String kCFStringEncodingUTF8); CFURLRef url = CFURLCreateWithFileSystemPath(NULL, path, kCFURLPOSIXPathStyle, NO); / / / get the PDF CGPDFDocumentRef document = CGPDFDocumentCreateWithURL (url); // Release CFRelease(path); CFRelease(url);Copy the code
  2. Get PDF page count

    NSInteger totoalPages = CGPDFDocumentGetNumberOfPages(document);
    Copy the code
  3. Get page count content

    CGPDFPageRef page = CGPDFDocumentGetPage(document, _currentPage);
    Copy the code
  4. Gets the current page size

    CGRect rect = CGPDFPageGetBoxRect(page, kCGPDFMediaBox);
    Copy the code
  5. Draws the current page content

    - (void)drawRect:(CGRect)rect { CGContextRef context = UIGraphicsGetCurrentContext(); [[UIColor whiteColor] set]; CGContextFillRect(context, rect); CGContextTranslateCTM(context, 0.0, rect.sie.height); CGContextScaleCTM (context, 1.0, 1.0); CGPDFPageRef page = CGPDFDocumentGetPage(document, _currentPage); CGAffineTransform pdfTransform = CGPDFPageGetDrawingTransform(page, kCGPDFCropBox, rect, 0, true); CGContextConcatCTM(context, pdfTransform); CGContextDrawPDFPage(context, page); }Copy the code
  6. The last step adds the page turn animation

    / / / return back [self transitionWithType: @ "pageUnCurl" subtype: kCATransitionFromRight]; / / / the next [self transitionWithType: @ "pageCurl" subtype: kCATransitionFromRight];Copy the code
    - (void)transitionWithType:(NSString *)type subtype:(CATransitionSubtype)subType { CATransition *animation = [CATransition animation]; Animation. duration = 0.8; /*** Use CATransitionType kCATransitionPush to push in effect kCATransitionMoveIn To move in effect kCATransitionReveal to cut off effect KCATransitionFade or simply use the following string: SuckEffect rippleEffect pageCurl up page turn pageUnCurl Down page turn oglFlip Up and down cameraIrisHollowOpen Camera shutter open CameraIrisHollowClose */ animation. Type = type; /// Set the animation direction /** kCATransitionFromRight from right kCATransitionFromLeft from left kCATransitionFromTop from above kCATransitionFromBottom from below  */ if (subType) { animation.subtype = subType; } /// the speed of animation /** CAMediaTimingFunction: KCAMediaTimingFunctionLinear uniform kCAMediaTimingFunctionEaseIn slowly into fast out kCAMediaTimingFunctionEaseOut fast forward slowly KCAMediaTimingFunctionEaseInEaseOut slow in slow out intermediate accelerating kCAMediaTimingFunctionDefault default * / animation timingFunction = [CAMediaTimingFunction functionWithName: kCAMediaTimingFunctionEaseIn]; [self.layer addAnimation:animation forKey:@"animation"]; }Copy the code

    A simple Demo is attached