Objective-C Issues with UIWebView to PDF

前端 未结 2 1520
日久生厌
日久生厌 2021-01-05 15:52

I have this method here that takes my UIWebView and convert into a PDF and its working well. But when I print off this PDF or email it, its cut off. Its like its only genera

相关标签:
2条回答
  • 2021-01-05 16:47

    I've done this in the past using UIPrintPageRenderer. It's a more versetile way of creating a PDF from a UIWebView, and it's been working well for me so far. I've tested this solution with Xcode 6 and iOS 8.2. Also, tried printing the resulting PDF and everything printed out fine.

    When I read the OP, I did some testing with various page sizes, to see if I can get a blank PDF. There are a few key items that I identified, that could contribute to a blank PDF file. I've identified them in the code.

    When webViewDidFinishLoad() gets called, the view might not be 100% loaded. A check is necessary, to see if the view is still loading. This is important, as it might be the source of your problem. If it's not, then we are good to go. There is a very important note here. Some web pages are loaded dynamically (defined in the page itself). Take youtube.com for example. The page displays almost immediately, with a "loading" screen. This will trick our web view, and it's "isLoading" property will be set to "false", while the web page is still loading content dynamically. This is a pretty rare case though, and in the general case this solution will work well. If you need to generate a PDF file from such a dynamic loading web page, you might need to move the actual generation to a different spot. Even with a dynamic loading web page, you will end up with a PDF showing the loading screen, and not an empty PDF file.

    Another key aspect is setting the printableRect and pageRect. Note that those are set separately. If the printableRect is smaller than the paperRect, you will end up with some padding around the content - see code for example. Here is a link to Apple's API doc with some short descriptions for both.

    The example code below adds a Category to UIPrintPageRenderer to create the actual PDF data. The code in this sample has been put together using various resources online in the past, and I wasn't able to find which ones were used to credit them properly.

    @interface UIPrintPageRenderer (PDF)
    - (NSData*) createPDF;
    @end
    
    @implementation UIPrintPageRenderer (PDF)
    - (NSData*) createPDF
    {
        NSMutableData *pdfData = [NSMutableData data];
        UIGraphicsBeginPDFContextToData( pdfData, self.paperRect, nil );
        [self prepareForDrawingPages: NSMakeRange(0, self.numberOfPages)];
        CGRect bounds = UIGraphicsGetPDFContextBounds();
        for ( int i = 0 ; i < self.numberOfPages ; i++ )
        {
            UIGraphicsBeginPDFPage();
            [self drawPageAtIndex: i inRect: bounds];
        }
        UIGraphicsEndPDFContext();
        return pdfData;
    }
    @end
    

    And here is what I have in webViewDidFinishLoad()

    - (void)webViewDidFinishLoad:(UIWebView *)webViewIn {
        NSLog(@"web view did finish loading");
    
        // webViewDidFinishLoad() could get called multiple times before
        // the page is 100% loaded. That's why we check if the page is still loading
        if (webViewIn.isLoading)
            return;
    
        UIPrintPageRenderer *render = [[UIPrintPageRenderer alloc] init];
        [render addPrintFormatter:webViewIn.viewPrintFormatter startingAtPageAtIndex:0];
    
        // Padding is desirable, but optional
        float padding = 10.0f;
    
        // Define the printableRect and paperRect
        // If the printableRect defines the printable area of the page
        CGRect paperRect = CGRectMake(0, 0, PDFSize.width, PDFSize.height);
        CGRect printableRect = CGRectMake(padding, padding, PDFSize.width-(padding * 2), PDFSize.height-(padding * 2));
    
        [render setValue:[NSValue valueWithCGRect:paperRect] forKey:@"paperRect"];
        [render setValue:[NSValue valueWithCGRect:printableRect] forKey:@"printableRect"];
    
        // Call the printToPDF helper method that will do the actual PDF creation using values set above
        NSData *pdfData = [render createPDF];
    
        // Save the PDF to a file, if creating one is successful
        if (pdfData) {
            NSArray *paths = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES);
            NSString *path = [paths objectAtIndex:0];
    
            NSString *pdfPath = [path stringByAppendingPathComponent:[NSString stringWithFormat:@"Purchase Order.pdf"]];
    
            [pdfData writeToFile:pdfPath atomically:YES];
        }
        else
        {
            NSLog(@"error creating PDF");
        }
    }
    

    PDFSize is defined as a constant, set to a standard A4 page size. It can be edited to meet your needs.

    #define PDFSize CGSizeMake(595.2,841.8)
    
    0 讨论(0)
  • 2021-01-05 16:52

    In order to create your PDF file in memory, you need to draw the layer of the UIWebBrowserView instance that lies underneath the UIWebView's scrollView. In order to do that, try changing your renderInContext: call the following way :

    UIView* contentView = webViewPDF.scrollView.subviews.firstObject;
    [contentView.layer renderInContext:currentContext];
    

    Also, if you target iOS >= 7.0, then you can avoid using renderInContext: and use one of the snapshotViewAfterScreenUpdates: or drawViewHierarchyInRect:afterScreenUpdates: methods.

    0 讨论(0)
提交回复
热议问题