UIWebView到PDF的Objective-C问题

use*_*331 6 pdf objective-c uiwebview ios

我在这里有这个方法,它将我的UIWebView转换成PDF并且运行良好.但是,当我打印此PDF或通过电子邮件发送时,它会被切断.它只是生成我设置的UIWebView的大小(宽度:688和高度:577)如果我增加UIWebView的大小,比如900或1024我的PDF是空的.我的UIWebView大于577,但在我的应用程序中,我能够滚动.

这是方法....

-(void)webViewDidFinishLoad:(UIWebView *)webViewPDF
{
    CGRect origframe = webViewPDF.frame;
    NSString *heightStr = [webViewPDF stringByEvaluatingJavaScriptFromString:@"document.body.scrollHeight;"]; // Get the height of our webView
    int height = [heightStr intValue];

    CGFloat maxHeight   = kDefaultPageHeight - 2*kMargin;
    int pages = floor(height / maxHeight);

    NSArray *paths = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES);
    NSString *path = [paths objectAtIndex:0];

    self.pdfPath = [path stringByAppendingPathComponent:[NSString stringWithFormat:@"Purchase Order.pdf"]];

    UIGraphicsBeginPDFContextToFile(self.pdfPath, CGRectZero, nil);

    for (int i = 0; i < pages; i++)
    {
        if (maxHeight * (i+1) > height) {
            CGRect f = [webViewPDF frame];
            f.size.height -= (((i+1) * maxHeight) - height);
            [webViewPDF setFrame: f];
        }

        UIGraphicsBeginPDFPageWithInfo(CGRectMake(0, 0, kDefaultPageWidth, kDefaultPageHeight), nil);
        CGContextRef currentContext = UIGraphicsGetCurrentContext();

        CGContextTranslateCTM(currentContext, kMargin, kMargin);

        [webViewPDF.layer renderInContext:currentContext];
    }

    UIGraphicsEndPDFContext();

    [webViewPDF setFrame:origframe];
    [[[webViewPDF subviews] lastObject] setContentOffset:CGPointMake(0, 0) animated:NO];
}
Run Code Online (Sandbox Code Playgroud)

我希望这是有道理的......有没有人对如何解决这个问题有任何建议,所以PDF没有被切断?

我忘了提到这些变量:

#define kDefaultPageHeight 850
#define kDefaultPageWidth  850
#define kMargin 50
Run Code Online (Sandbox Code Playgroud)

这是我的分享按钮:

- (IBAction)Share:(id)sender {

    NSData *pdfData = [NSData dataWithContentsOfFile:self.pdfPath];

    UIActivityViewController * activityController = [[UIActivityViewController alloc] initWithActivityItems:@[pdfData] applicationActivities:nil];

    UIPopoverController *popup = [[UIPopoverController alloc] initWithContentViewController:activityController];

    [popup presentPopoverFromRect:CGRectMake(self.view.frame.size.width - 36, 60, 0, 0)inView:self.view permittedArrowDirections:UIPopoverArrowDirectionUp animated:YES];

}
Run Code Online (Sandbox Code Playgroud)

Vel*_*nov 5

我以前使用UIPrintPageRenderer完成了这个.这是一种从UIWebView创建PDF的更通用的方式,到目前为止它一直很适合我.我用Xcode 6和iOS 8.2测试了这个解决方案.此外,尝试打印生成的PDF和打印出来的一切都很好.

当我阅读OP时,我做了一些不同页面大小的测试,看看我是否可以得到一个空白的PDF.我发现了一些关键项,可能会导致空白的PDF文件.我已在代码中识别出它们.

调用webViewDidFinishLoad()时,视图可能不会100%加载.必须进行检查,以查看视图是否仍在加载.这很重要,因为它可能是您问题的根源.如果不是,那么我们很高兴.这里有一个非常重要的注意事项.一些网页是动态加载的(在页面本身中定义).以youtube.com为例.页面几乎立即显示,并带有"加载"屏幕.这将欺骗我们的Web视图,并且它的"isLoading"属性将设置为"false",而网页仍然会动态加载内容.这是一个非常罕见的情况,在一般情况下,这个解决方案将运作良好.如果需要从这样的动态加载网页生成PDF文件,则可能需要将实际生成移动到其他位置.即使使用动态加载网页,您最终也会看到显示加载屏幕的PDF,而不是空PDF文件.

另一个关键方面是设置printableRect和pageRect.请注意,这些是单独设置的.如果printableRect小于paperRect,则最终会在内容周围添加一些填充 - 例如,请参阅代码.以下是Apple API文档的链接,其中包含两个简短描述.

下面的示例代码将类别添加到UIPrintPageRenderer以创建实际的PDF数据.这个示例中的代码在过去使用各种在线资源放在一起,我无法找到哪些资源被用于正确地归功于它们.

@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
Run Code Online (Sandbox Code Playgroud)

这是我在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");
    }
}
Run Code Online (Sandbox Code Playgroud)

PDFSize定义为常量,设置为标准A4页面大小.它可以编辑以满足您的需求.

#define PDFSize CGSizeMake(595.2,841.8)
Run Code Online (Sandbox Code Playgroud)