Поиск гиперссылок в PDF с помощью CGPDF - PullRequest
3 голосов
/ 26 июля 2011

Я пытаюсь найти все гиперссылки на странице в документе PDF и поместить поверх них кнопку UIB.

Я нашел и поиграл со сценарием ниже, который должен быть в состоянии это сделать. Он находит гиперссылки и размещает кнопки UIB, но их нет над ссылками. Может кто-нибудь сказать мне, почему это может быть?

Я приложил изображение, чтобы показать, как кнопки выключены. Самые верхние кнопки должны быть сверху синего текста «METROPOLITAN», а самые нижние кнопки должны быть сверху текста с надписью «SÆSONEN 2008-2009».

Это код, который я использую, чтобы найти ссылки и разместить кнопки.

    NSMutableArray *rectArrays;
    NSMutableArray *storeURLs;

    CGPDFPageRef pdfPage = CGPDFDocumentGetPage(_pdf, pageNumber);
    CGPDFDictionaryRef pageDictionary = CGPDFPageGetDictionary(pdfPage);

    CGPDFArrayRef outputArray;
    if(!CGPDFDictionaryGetArray(pageDictionary, "Annots", &outputArray)) {
        //break;
    }

    int arrayCount = 0;
    arrayCount = CGPDFArrayGetCount(outputArray);
    if(arrayCount > 0) {
        for(int j = 0; j < arrayCount; ++j) {
            CGPDFObjectRef aDictObj;
            if(!CGPDFArrayGetObject(outputArray, j, &aDictObj)) {
                return;
            }

            CGPDFDictionaryRef annotDict;
            if(!CGPDFObjectGetValue(aDictObj, kCGPDFObjectTypeDictionary, &annotDict)) {
                return;
            }

            CGPDFDictionaryRef aDict;
            if(!CGPDFDictionaryGetDictionary(annotDict, "A", &aDict)) {
                return;
            }

            CGPDFStringRef uriStringRef;
            if(!CGPDFDictionaryGetString(aDict, "URI", &uriStringRef)) {
                return;
            }

            CGPDFArrayRef rectArray;
            if(!CGPDFDictionaryGetArray(annotDict, "Rect", &rectArray)) {
                return;
            }

            int arrayCount = CGPDFArrayGetCount( rectArray );
            CGPDFReal coords[4];
            for(int k = 0; k < arrayCount; ++k) {
                CGPDFObjectRef rectObj;
                if(!CGPDFArrayGetObject(rectArray, k, &rectObj)) {
                    return;
                }

                CGPDFReal coord;
                if(!CGPDFObjectGetValue(rectObj, kCGPDFObjectTypeReal, &coord)) {
                    return;
                }

                coords[k] = coord;
            }

            char *uriString = (char *)CGPDFStringGetBytePtr(uriStringRef);

            NSString *uri = [NSString stringWithCString:uriString encoding:NSUTF8StringEncoding];

            CGRect rect = CGRectMake(coords[0], coords[1], coords[2], coords[3]);

            NSLog(@"Found: %f ; %f => %f x %f", coords[0], coords[1], coords[2], coords[3]);

            CGPDFInteger pageRotate = 0;
            CGPDFDictionaryGetInteger(pageDictionary, "Rotate", &pageRotate); 
            CGRect pageRect = CGRectIntegral( CGPDFPageGetBoxRect(pdfPage, kCGPDFMediaBox));
            if(pageRotate == 90 || pageRotate == 270) {
                CGFloat temp = pageRect.size.width;
                pageRect.size.width = pageRect.size.height;
                pageRect.size.height = temp;
            }

            rect.size.width = rect.size.width - rect.origin.x;
            rect.size.height = rect.size.height - rect.origin.y;

            CGAffineTransform trans = CGAffineTransformIdentity;
            trans = CGAffineTransformTranslate(trans, 0, pageRect.size.height);
            trans = CGAffineTransformScale(trans, 1.0, -1.0);

            rect = CGRectApplyAffineTransform(rect, trans);

            NSURL *url = [NSURL URLWithString:uri];
            [rectArrays addObject:[NSValue valueWithCGRect:rect]];
            [storeURLs addObject:url];

            for(int i = 0; i <= [rectArrays count]; i++) {
                UIButton *button = [UIButton buttonWithType:UIButtonTypeRoundedRect];
                [button setFrame:rect];
                [button setTitle:@"Link" forState:UIControlStateNormal];
                [[self view] addSubview:button];

                NSLog(@"Added: %f ; %f => %f x %f", button.frame.origin.x, button.frame.origin.y, button.frame.size.width, button.frame.size.height);
            }
        }
    }

enter image description here

РЕДАКТИРОВАТЬ : Попытка настроить коэффициент масштабирования, но он не работает. Кнопки UIB по-прежнему расположены неправильно.

CGAffineTransform trans = CGAffineTransformIdentity;
trans = CGAffineTransformTranslate(trans, 0, pageRect.size.height * 1.5);
trans = CGAffineTransformScale(trans, 1.5, -1.5);

РЕДАКТИРОВАТЬ Я взглянул на исходный код, который iPDFDev также связывает, и кажется, что у меня сейчас работает позиционирование. Мои кнопки теперь слишком длинные. Кто-нибудь знает, почему это может быть? Если вычесть viewRext.origin.x из viewRect.size.width, кнопки слишком короткие.

Теперь это мой код.

    CGPDFPageRef pdfPage = CGPDFDocumentGetPage(_pdf, pageNumber);
    CGPDFDictionaryRef pageDictionary = CGPDFPageGetDictionary(pdfPage);

    CGPDFArrayRef outputArray;
    if(!CGPDFDictionaryGetArray(pageDictionary, "Annots", &outputArray)) {
        //break;
    }

    int arrayCount = 0;
    arrayCount = CGPDFArrayGetCount(outputArray);
    if(arrayCount > 0) {
        for(int j = 0; j < arrayCount; ++j) {
            CGPDFObjectRef aDictObj;
            if(!CGPDFArrayGetObject(outputArray, j, &aDictObj)) {
                return;
            }

            CGPDFDictionaryRef annotDict;
            if(!CGPDFObjectGetValue(aDictObj, kCGPDFObjectTypeDictionary, &annotDict)) {
                return;
            }

            CGPDFDictionaryRef aDict;
            if(!CGPDFDictionaryGetDictionary(annotDict, "A", &aDict)) {
                return;
            }

            CGPDFStringRef uriStringRef;
            if(!CGPDFDictionaryGetString(aDict, "URI", &uriStringRef)) {
                return;
            }

            CGPDFArrayRef rectArray;
            if(!CGPDFDictionaryGetArray(annotDict, "Rect", &rectArray)) {
                return;
            }

            int arrayCount = CGPDFArrayGetCount( rectArray );
            CGPDFReal coords[4];
            for(int k = 0; k < arrayCount; ++k) {
                CGPDFObjectRef rectObj;
                if(!CGPDFArrayGetObject(rectArray, k, &rectObj)) {
                    return;
                }

                CGPDFReal coord;
                if(!CGPDFObjectGetValue(rectObj, kCGPDFObjectTypeReal, &coord)) {
                    return;
                }

                coords[k] = coord;
            }

            char *uriString = (char *)CGPDFStringGetBytePtr(uriStringRef);

            NSString *uri = [NSString stringWithCString:uriString encoding:NSUTF8StringEncoding];

            CGRect rect = CGRectMake(coords[0], coords[1], coords[2], coords[3]);

            CGPoint pdfPoint = CGPointMake(rect.origin.x, rect.origin.y);
            CGPoint viewPoint = CGPointMake(0, 0);

            CGRect cropBox = CGPDFPageGetBoxRect(pdfPage, kCGPDFCropBox);

            int rotation = CGPDFPageGetRotationAngle(pdfPage);

            CGRect pageRenderRect;
            switch (rotation) {
                case 90:
                case -270:
                    pageRenderRect = CGRectMake(0, 0, 1024, 768);

                    viewPoint.x = pageRenderRect.size.width * (pdfPoint.y - cropBox.origin.y) / cropBox.size.height;
                    viewPoint.y = pageRenderRect.size.height * (pdfPoint.x - cropBox.origin.x) / cropBox.size.width;
                    break;
                case 180:
                case -180:
                    pageRenderRect = CGRectMake(0, 0, 768, 1024);

                    viewPoint.x = pageRenderRect.size.width * (cropBox.size.width - (pdfPoint.x - cropBox.origin.x)) / cropBox.size.width;
                    viewPoint.y = pageRenderRect.size.height * (pdfPoint.y - cropBox.origin.y) / cropBox.size.height;
                    break;
                case -90:
                case 270:
                    pageRenderRect = CGRectMake(0, 0, 1024, 768);

                    viewPoint.x = pageRenderRect.size.width * (cropBox.size.height - (pdfPoint.y - cropBox.origin.y)) / cropBox.size.height;
                    viewPoint.y = pageRenderRect.size.height * (cropBox.size.width - (pdfPoint.x - cropBox.origin.x)) / cropBox.size.width;
                    break;
                case 0:
                default:
                    pageRenderRect = CGRectMake(0, 0, 768, 1024);

                    viewPoint.x = pageRenderRect.size.width * (pdfPoint.x - cropBox.origin.x) / cropBox.size.width;
                    viewPoint.y = pageRenderRect.size.height * (cropBox.size.height - pdfPoint.y) / cropBox.size.height;
                    break;
            }

            viewPoint.x = viewPoint.x + pageRenderRect.origin.x;
            viewPoint.y = viewPoint.y + pageRenderRect.origin.y;

            CGRect viewRect = CGRectMake(viewPoint.x, viewPoint.y, rect.size.width, rect.size.height - rect.origin.y);
            viewRect = CGRectMake(viewRect.origin.x, viewRect.origin.y - viewRect.size.height, viewRect.size.width, viewRect.size.height);

            UIButton *button = [UIButton buttonWithType:UIButtonTypeCustom];
            [button setFrame:viewRect];
            [button setBackgroundColor:[UIColor greenColor]];
            [button setAlpha:0.65];
            [button setTag:kPDFLinkButton];
            [[self view] addSubview:button];
        }
    }

РЕДАКТИРОВАТЬ: Это мой окончательный код. См. Ответ от iPDFDev для получения дополнительной информации.

- (void)getLinksFromPDF:(CGPDFDocumentRef)_pdf withPageNumber:(NSUInteger)pageNumber
{
    CGPDFPageRef pdfPage = CGPDFDocumentGetPage(_pdf, pageNumber);
    CGPDFDictionaryRef pageDictionary = CGPDFPageGetDictionary(pdfPage);

    CGPDFArrayRef outputArray;
    if(!CGPDFDictionaryGetArray(pageDictionary, "Annots", &outputArray)) {
        //break;
    }

    int arrayCount = 0;
    arrayCount = CGPDFArrayGetCount(outputArray);
    if(arrayCount > 0) {
        for(int j = 0; j < arrayCount; ++j) {
            CGPDFObjectRef aDictObj;
            if(!CGPDFArrayGetObject(outputArray, j, &aDictObj)) {
                return;
            }

            CGPDFDictionaryRef annotDict;
            if(!CGPDFObjectGetValue(aDictObj, kCGPDFObjectTypeDictionary, &annotDict)) {
                return;
            }

            CGPDFDictionaryRef aDict;
            if(!CGPDFDictionaryGetDictionary(annotDict, "A", &aDict)) {
                return;
            }

            CGPDFStringRef uriStringRef;
            if(!CGPDFDictionaryGetString(aDict, "URI", &uriStringRef)) {
                return;
            }

            CGPDFArrayRef rectArray;
            if(!CGPDFDictionaryGetArray(annotDict, "Rect", &rectArray)) {
                return;
            }

            int arrayCount = CGPDFArrayGetCount( rectArray );
            CGPDFReal coords[4];
            for(int k = 0; k < arrayCount; ++k) {
                CGPDFObjectRef rectObj;
                if(!CGPDFArrayGetObject(rectArray, k, &rectObj)) {
                    return;
                }

                CGPDFReal coord;
                if(!CGPDFObjectGetValue(rectObj, kCGPDFObjectTypeReal, &coord)) {
                    return;
                }

                coords[k] = coord;
            }

            char *uriString = (char *)CGPDFStringGetBytePtr(uriStringRef);

            NSString *uri = [NSString stringWithCString:uriString encoding:NSUTF8StringEncoding];

            CGPoint lowerLeft = [self convertPDFPointToViewPoint:CGPointMake(coords[0], coords[1])];
            CGPoint upperRight = [self convertPDFPointToViewPoint:CGPointMake(coords[2], coords[3])];

            // This is the rectangle positioned under the link
            CGRect viewRect = CGRectMake(lowerLeft.x, lowerLeft.y, upperRight.x - lowerLeft.x, lowerLeft.y - upperRight.y);

            // Now adjusting the rectangle to be on top of the link
            viewRect = CGRectMake(viewRect.origin.x, viewRect.origin.y - viewRect.size.height, viewRect.size.width, viewRect.size.height);

            NSLog(@"%@", uri);

            UIButton *button = [UIButton buttonWithType:UIButtonTypeCustom];
            [button setFrame:viewRect];
            [button setBackgroundColor:[UIColor greenColor]];
            [button setAlpha:0.65];
            [button setTag:kPDFLinkButton];
            [[self view] addSubview:button];
        }
    }   
}

- (CGPoint)convertPDFPointToViewPoint:(CGPoint)pdfPoint
{
    CGPoint viewPoint = CGPointMake(0, 0);

    CGPDFPageRef pdfPage = CGPDFDocumentGetPage(pdf, currentPage);

    CGRect cropBox = CGPDFPageGetBoxRect(pdfPage, kCGPDFCropBox);

    int rotation = CGPDFPageGetRotationAngle(pdfPage);

    CGRect pageRenderRect;
    switch (rotation) {
        case 90:
        case -270:
            pageRenderRect = CGRectMake(0, 0, 1024, 768);

            viewPoint.x = pageRenderRect.size.width * (pdfPoint.y - cropBox.origin.y) / cropBox.size.height;
            viewPoint.y = pageRenderRect.size.height * (pdfPoint.x - cropBox.origin.x) / cropBox.size.width;
            break;
        case 180:
        case -180:
            pageRenderRect = CGRectMake(0, 0, 768, 1024);

            viewPoint.x = pageRenderRect.size.width * (cropBox.size.width - (pdfPoint.x - cropBox.origin.x)) / cropBox.size.width;
            viewPoint.y = pageRenderRect.size.height * (pdfPoint.y - cropBox.origin.y) / cropBox.size.height;
            break;
        case -90:
        case 270:
            pageRenderRect = CGRectMake(0, 0, 1024, 768);

            viewPoint.x = pageRenderRect.size.width * (cropBox.size.height - (pdfPoint.y - cropBox.origin.y)) / cropBox.size.height;
            viewPoint.y = pageRenderRect.size.height * (cropBox.size.width - (pdfPoint.x - cropBox.origin.x)) / cropBox.size.width;
            break;
        case 0:
        default:
            pageRenderRect = CGRectMake(0, 0, 768, 1024);

            viewPoint.x = pageRenderRect.size.width * (pdfPoint.x - cropBox.origin.x) / cropBox.size.width;
            viewPoint.y = pageRenderRect.size.height * (cropBox.size.height - pdfPoint.y) / cropBox.size.height;
            break;
    }

    viewPoint.x = viewPoint.x + pageRenderRect.origin.x;
    viewPoint.y = viewPoint.y + pageRenderRect.origin.y;

    return viewPoint;
}

Ответы [ 2 ]

4 голосов
/ 26 июля 2011

Ссылки хранятся в массиве Annots в словаре страницы.Когда вы просматриваете массив Annots, вы обнаруживаете аннотации ссылок, проверяя ключ Subtype в словаре аннотаций, он должен быть «Link» (имя объекта).Если у вас есть аннотация ссылки, вы проверяете, есть ли у нее действие, сохраненное в записи A.Если у него есть действие, вы проверяете, является ли это веб-ссылкой, запись S в словаре действий должна иметь значение «URI» (имя объекта).Если у вас есть действие URI, вы извлекаете ссылку из записи URI в словаре действий.

Позиция аннотации на странице PDF сохраняется в массиве Rect.Это 4 значения для 2 противоположных углов, обычно нижний левый x, нижний левый y, верхний правый x, верхний правый y.Эти значения относятся к левому нижнему углу медиа-блока страницы.Эти значения должны быть преобразованы в координаты вида на основе следующих переменных: 1. позиция в представлении, где отображается страница;2. размер отображаемой страницы (текущий коэффициент масштабирования * размер страницы PDF);3. окно обрезки страницы;4. поворот страницы.

Метод преобразования выглядит следующим образом:

- (CGPoint)convertPDFPointToViewPoint:(CGPoint)pdfPoint {
    CGPoint viewPoint = CGPointMake(0, 0);

    CGRect cropBox = CGPDFPageGetBoxRect(pdfPage, kCGPDFCropBox);

    int rotation = CGPDFPageGetRotationAngle(pdfPage);

    switch (rotation) {
        case 90:
        case -270:
            viewPoint.x = pageRenderRect.size.width * (pdfPoint.y - cropBox.origin.y) / cropBox.size.height;
            viewPoint.y = pageRenderRect.size.height * (pdfPoint.x - cropBox.origin.x) / cropBox.size.width;
            break;
        case 180:
        case -180:
            viewPoint.x = pageRenderRect.size.width * (cropBox.size.width - (pdfPoint.x - cropBox.origin.x)) / cropBox.size.width;
            viewPoint.y = pageRenderRect.size.height * (pdfPoint.y - cropBox.origin.y) / cropBox.size.height;
            break;
        case -90:
        case 270:
            viewPoint.x = pageRenderRect.size.width * (cropBox.size.height - (pdfPoint.y - cropBox.origin.y)) / cropBox.size.height;
            viewPoint.y = pageRenderRect.size.height * (cropBox.size.width - (pdfPoint.x - cropBox.origin.x)) / cropBox.size.width;
            break;
        case 0:
        default:
            viewPoint.x = pageRenderRect.size.width * (pdfPoint.x - cropBox.origin.x) / cropBox.size.width;
            viewPoint.y = pageRenderRect.size.height * (cropBox.size.height - pdfPoint.y) / cropBox.size.height;
            break;
    }

    viewPoint.x = viewPoint.x + pageRenderRect.origin.x;
    viewPoint.y = viewPoint.y + pageRenderRect.origin.y;

    return viewPoint;
}

, где pageRenderRect - это прямоугольник в представлении, где отображается страница.Вы можете найти дополнительный справочный исходный код здесь: http://ipdfdev.com/2011/06/21/links-navigation-in-a-pdf-document-on-iphone-and-ipad/

1 голос
/ 26 июля 2011

Предположительно, возможно, вы применяете какое-то вычисленное преобразование для масштабирования содержимого PDF до любого имеющегося кадра, но не можете применить его к рамкам кнопок?Вы, кажется, жестко закодировали преобразование в показанный код, который предполагает, что PDF будет отображаться с одной исходной точкой, равной одной точке назначения, если вы вообще масштабируете PDF, тогда, очевидно, это не будет иметь место.

Этот диагноз объясняет, почему похоже, что эти кнопки предназначены для обозначения 'Metropolitan' и 'Saesonen 2008-2009', но их система координат была уменьшена до верхнего левого угла.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...