【问题标题】:Objective-C Issues with UIWebView to PDFUIWebView 到 PDF 的 Objective-C 问题
【发布时间】:2015-07-23 23:34:15
【问题描述】:

我这里有这个方法,它可以将我的 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];
}

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

我忘了提到这些变量:

#define kDefaultPageHeight 850
#define kDefaultPageWidth  850
#define kMargin 50

这是我的分享按钮:

- (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];

}

【问题讨论】:

  • 那么这和Objective-C有什么关系呢?
  • 这个方法是Objective-C
  • 显然您的代码恰好在 Objective-C 中。任何人都可以看到这一点。但问题不在于关于 Objective-C。无论您使用的是 Objective-C、Swift、AppleScript 还是其他任何东西,答案都是一样的。
  • 您是否尝试过将 UIWebview 的边界设置为 PDF 中页面的大小,然后滚动 UIWebview 以便在调用 layer.renderInContext 之前显示该页面所需的确切片段? UIWebview 可能会优化绘图,使其仅将在任何给定时间可见的内容部分绘制到其图层。
  • 我不明白@DavidvanDriessche,你能回答一下吗?

标签: ios objective-c pdf uiwebview


【解决方案1】:

我过去曾使用 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 文档的link,其中包含两者的简短描述。

下面的示例代码向 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

这就是我在 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 定义为常量,设置为标准 A4 页面大小。可以对其进行编辑以满足您的需求。

#define PDFSize CGSizeMake(595.2,841.8)

【讨论】:

  • 我收到此代码错误:未定义 PDFSize。它应该是什么。此外,当我按下我的分享按钮时,应用程序崩溃attempt to insert nil object from objects[0] 我已经用我的分享按钮更新了我的问题
  • PDFSize 是一个常数。它基本上定义了您的页面大小。我会在这里使用一些标准的东西,这就是我将它设置为 A4 的原因。你可以修改它以获得你需要的页面大小。
  • 就您的 share() 操作而言,当 PDF 在您定义的路径中不可用时,您会遇到问题。您可以尝试几件事 - 确保使用相同的路径进行保存和共享文件。另一件事,这一点更重要,确保 PDF 文件实际创建并在该路径中可用。我最终测试了该操作,但没有发生崩溃。
  • @VelGenov :我正在以与 Microsoft 文档相同的方式创建 PDF。但是在这里,创建的 PDF 具有灰色背景。可以改吗?
  • @MansiPanchal,这个线程专门用于在 iOS 中查看 PDF。在 Microsoft 环境中创建 PDF 时,您可能会看到一个单独的问题。您将不得不为此发布一个单独的问题并添加一些详细信息。谢谢。
【解决方案2】:

为了在内存中创建 PDF 文件,您需要绘制位于 UIWebViewscrollView 下方的 UIWebBrowserView 实例层。为此,请尝试通过以下方式更改您的 renderInContext: 调用:

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

另外,如果您的目标是 iOS >= 7.0,那么您可以避免使用 renderInContext: 并使用 snapshotViewAfterScreenUpdates: 之一 或drawViewHierarchyInRect:afterScreenUpdates: 方法。

【讨论】:

  • 我应该把它放在我的方法中的什么地方?
  • 替换你已有的电话:[webViewPDF.layer renderInContext:currentContext];
  • 你是我的上帝!谢谢!
猜你喜欢
  • 1970-01-01
  • 2011-02-21
  • 2015-03-06
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多