| // Copyright 2014 PDFium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| // Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com |
| |
| #include "public/fpdf_edit.h" |
| |
| #include "core/fpdfapi/cpdf_modulemgr.h" |
| #include "core/fpdfapi/page/cpdf_image.h" |
| #include "core/fpdfapi/page/cpdf_imageobject.h" |
| #include "core/fpdfapi/page/cpdf_page.h" |
| #include "core/fpdfapi/page/cpdf_pageobject.h" |
| #include "core/fpdfapi/parser/cpdf_array.h" |
| #include "core/fpdfapi/parser/cpdf_name.h" |
| #include "core/fpdfapi/render/cpdf_dibsource.h" |
| #include "fpdfsdk/fsdk_define.h" |
| #include "third_party/base/ptr_util.h" |
| |
| namespace { |
| |
| // These checks ensure the consistency of colorspace values across core/ and |
| // public/. |
| static_assert(PDFCS_DEVICEGRAY == FPDF_COLORSPACE_DEVICEGRAY, |
| "PDFCS_DEVICEGRAY value mismatch"); |
| static_assert(PDFCS_DEVICERGB == FPDF_COLORSPACE_DEVICERGB, |
| "PDFCS_DEVICERGB value mismatch"); |
| static_assert(PDFCS_DEVICECMYK == FPDF_COLORSPACE_DEVICECMYK, |
| "PDFCS_DEVICECMYK value mismatch"); |
| static_assert(PDFCS_CALGRAY == FPDF_COLORSPACE_CALGRAY, |
| "PDFCS_CALGRAY value mismatch"); |
| static_assert(PDFCS_CALRGB == FPDF_COLORSPACE_CALRGB, |
| "PDFCS_CALRGB value mismatch"); |
| static_assert(PDFCS_LAB == FPDF_COLORSPACE_LAB, "PDFCS_LAB value mismatch"); |
| static_assert(PDFCS_ICCBASED == FPDF_COLORSPACE_ICCBASED, |
| "PDFCS_ICCBASED value mismatch"); |
| static_assert(PDFCS_SEPARATION == FPDF_COLORSPACE_SEPARATION, |
| "PDFCS_SEPARATION value mismatch"); |
| static_assert(PDFCS_DEVICEN == FPDF_COLORSPACE_DEVICEN, |
| "PDFCS_DEVICEN value mismatch"); |
| static_assert(PDFCS_INDEXED == FPDF_COLORSPACE_INDEXED, |
| "PDFCS_INDEXED value mismatch"); |
| static_assert(PDFCS_PATTERN == FPDF_COLORSPACE_PATTERN, |
| "PDFCS_PATTERN value mismatch"); |
| |
| bool LoadJpegHelper(FPDF_PAGE* pages, |
| int nCount, |
| FPDF_PAGEOBJECT image_object, |
| FPDF_FILEACCESS* fileAccess, |
| bool inlineJpeg) { |
| if (!image_object || !fileAccess) |
| return false; |
| |
| RetainPtr<IFX_SeekableReadStream> pFile = MakeSeekableReadStream(fileAccess); |
| CPDF_ImageObject* pImgObj = static_cast<CPDF_ImageObject*>(image_object); |
| |
| if (pages) { |
| for (int index = 0; index < nCount; index++) { |
| CPDF_Page* pPage = CPDFPageFromFPDFPage(pages[index]); |
| if (pPage) |
| pImgObj->GetImage()->ResetCache(pPage, nullptr); |
| } |
| } |
| |
| if (inlineJpeg) |
| pImgObj->GetImage()->SetJpegImageInline(pFile); |
| else |
| pImgObj->GetImage()->SetJpegImage(pFile); |
| pImgObj->SetDirty(true); |
| return true; |
| } |
| |
| } // namespace |
| |
| FPDF_EXPORT FPDF_PAGEOBJECT FPDF_CALLCONV |
| FPDFPageObj_NewImageObj(FPDF_DOCUMENT document) { |
| CPDF_Document* pDoc = CPDFDocumentFromFPDFDocument(document); |
| if (!pDoc) |
| return nullptr; |
| |
| auto pImageObj = pdfium::MakeUnique<CPDF_ImageObject>(); |
| pImageObj->SetImage(pdfium::MakeRetain<CPDF_Image>(pDoc)); |
| return pImageObj.release(); |
| } |
| |
| FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV |
| FPDFImageObj_LoadJpegFile(FPDF_PAGE* pages, |
| int nCount, |
| FPDF_PAGEOBJECT image_object, |
| FPDF_FILEACCESS* fileAccess) { |
| return LoadJpegHelper(pages, nCount, image_object, fileAccess, false); |
| } |
| |
| FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV |
| FPDFImageObj_LoadJpegFileInline(FPDF_PAGE* pages, |
| int nCount, |
| FPDF_PAGEOBJECT image_object, |
| FPDF_FILEACCESS* fileAccess) { |
| return LoadJpegHelper(pages, nCount, image_object, fileAccess, true); |
| } |
| |
| FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV |
| FPDFImageObj_SetMatrix(FPDF_PAGEOBJECT image_object, |
| double a, |
| double b, |
| double c, |
| double d, |
| double e, |
| double f) { |
| if (!image_object) |
| return false; |
| |
| CPDF_ImageObject* pImgObj = static_cast<CPDF_ImageObject*>(image_object); |
| pImgObj->set_matrix(CFX_Matrix(static_cast<float>(a), static_cast<float>(b), |
| static_cast<float>(c), static_cast<float>(d), |
| static_cast<float>(e), static_cast<float>(f))); |
| pImgObj->CalcBoundingBox(); |
| pImgObj->SetDirty(true); |
| return true; |
| } |
| |
| FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV |
| FPDFImageObj_SetBitmap(FPDF_PAGE* pages, |
| int nCount, |
| FPDF_PAGEOBJECT image_object, |
| FPDF_BITMAP bitmap) { |
| if (!image_object || !bitmap || !pages) |
| return false; |
| |
| CPDF_ImageObject* pImgObj = static_cast<CPDF_ImageObject*>(image_object); |
| for (int index = 0; index < nCount; index++) { |
| CPDF_Page* pPage = CPDFPageFromFPDFPage(pages[index]); |
| if (pPage) |
| pImgObj->GetImage()->ResetCache(pPage, nullptr); |
| } |
| RetainPtr<CFX_DIBitmap> holder(CFXBitmapFromFPDFBitmap(bitmap)); |
| pImgObj->GetImage()->SetImage(holder); |
| pImgObj->CalcBoundingBox(); |
| pImgObj->SetDirty(true); |
| return true; |
| } |
| |
| FPDF_EXPORT FPDF_BITMAP FPDF_CALLCONV |
| FPDFImageObj_GetBitmap(FPDF_PAGEOBJECT image_object) { |
| CPDF_PageObject* pObj = CPDFPageObjectFromFPDFPageObject(image_object); |
| if (!pObj || !pObj->IsImage()) |
| return nullptr; |
| |
| RetainPtr<CPDF_Image> pImg = pObj->AsImage()->GetImage(); |
| if (!pImg) |
| return nullptr; |
| |
| RetainPtr<CFX_DIBSource> pSource = pImg->LoadDIBSource(); |
| if (!pSource) |
| return nullptr; |
| |
| RetainPtr<CFX_DIBitmap> pBitmap; |
| // If the source image has a representation of 1 bit per pixel, then convert |
| // it to a grayscale bitmap having 1 byte per pixel, since bitmaps have no |
| // concept of bits. Otherwise, convert the source image to a bitmap directly, |
| // retaining its color representation. |
| if (pSource->GetBPP() == 1) |
| pBitmap = pSource->CloneConvert(FXDIB_8bppRgb); |
| else |
| pBitmap = pSource->Clone(nullptr); |
| |
| return pBitmap.Leak(); |
| } |
| |
| FPDF_EXPORT unsigned long FPDF_CALLCONV |
| FPDFImageObj_GetImageDataDecoded(FPDF_PAGEOBJECT image_object, |
| void* buffer, |
| unsigned long buflen) { |
| CPDF_PageObject* pObj = CPDFPageObjectFromFPDFPageObject(image_object); |
| if (!pObj || !pObj->IsImage()) |
| return 0; |
| |
| RetainPtr<CPDF_Image> pImg = pObj->AsImage()->GetImage(); |
| if (!pImg) |
| return 0; |
| |
| CPDF_Stream* pImgStream = pImg->GetStream(); |
| if (!pImgStream) |
| return 0; |
| |
| return DecodeStreamMaybeCopyAndReturnLength(pImgStream, buffer, buflen); |
| } |
| |
| FPDF_EXPORT unsigned long FPDF_CALLCONV |
| FPDFImageObj_GetImageDataRaw(FPDF_PAGEOBJECT image_object, |
| void* buffer, |
| unsigned long buflen) { |
| CPDF_PageObject* pObj = CPDFPageObjectFromFPDFPageObject(image_object); |
| if (!pObj || !pObj->IsImage()) |
| return 0; |
| |
| RetainPtr<CPDF_Image> pImg = pObj->AsImage()->GetImage(); |
| if (!pImg) |
| return 0; |
| |
| CPDF_Stream* pImgStream = pImg->GetStream(); |
| if (!pImgStream) |
| return 0; |
| |
| uint32_t len = pImgStream->GetRawSize(); |
| if (buffer && buflen >= len) |
| memcpy(buffer, pImgStream->GetRawData(), len); |
| |
| return len; |
| } |
| |
| FPDF_EXPORT int FPDF_CALLCONV |
| FPDFImageObj_GetImageFilterCount(FPDF_PAGEOBJECT image_object) { |
| CPDF_PageObject* pObj = CPDFPageObjectFromFPDFPageObject(image_object); |
| if (!pObj || !pObj->IsImage()) |
| return 0; |
| |
| RetainPtr<CPDF_Image> pImg = pObj->AsImage()->GetImage(); |
| if (!pImg) |
| return 0; |
| |
| CPDF_Dictionary* pDict = pImg->GetDict(); |
| CPDF_Object* pFilter = pDict ? pDict->GetDirectObjectFor("Filter") : nullptr; |
| if (!pFilter) |
| return 0; |
| |
| if (pFilter->IsArray()) |
| return pFilter->AsArray()->GetCount(); |
| if (pFilter->IsName()) |
| return 1; |
| |
| return 0; |
| } |
| |
| FPDF_EXPORT unsigned long FPDF_CALLCONV |
| FPDFImageObj_GetImageFilter(FPDF_PAGEOBJECT image_object, |
| int index, |
| void* buffer, |
| unsigned long buflen) { |
| if (index < 0 || index >= FPDFImageObj_GetImageFilterCount(image_object)) |
| return 0; |
| |
| CPDF_PageObject* pObj = CPDFPageObjectFromFPDFPageObject(image_object); |
| CPDF_Object* pFilter = |
| pObj->AsImage()->GetImage()->GetDict()->GetDirectObjectFor("Filter"); |
| ByteString bsFilter; |
| if (pFilter->IsName()) |
| bsFilter = pFilter->AsName()->GetString(); |
| else |
| bsFilter = pFilter->AsArray()->GetStringAt(index); |
| |
| unsigned long len = bsFilter.GetLength() + 1; |
| if (buffer && len <= buflen) |
| memcpy(buffer, bsFilter.c_str(), len); |
| return len; |
| } |
| |
| FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV |
| FPDFImageObj_GetImageMetadata(FPDF_PAGEOBJECT image_object, |
| FPDF_PAGE page, |
| FPDF_IMAGEOBJ_METADATA* metadata) { |
| CPDF_PageObject* pObj = CPDFPageObjectFromFPDFPageObject(image_object); |
| if (!pObj || !pObj->IsImage() || !metadata) |
| return false; |
| |
| RetainPtr<CPDF_Image> pImg = pObj->AsImage()->GetImage(); |
| if (!pImg) |
| return false; |
| |
| const int nPixelWidth = pImg->GetPixelWidth(); |
| const int nPixelHeight = pImg->GetPixelHeight(); |
| metadata->width = nPixelWidth; |
| metadata->height = nPixelHeight; |
| |
| const float nWidth = pObj->m_Right - pObj->m_Left; |
| const float nHeight = pObj->m_Top - pObj->m_Bottom; |
| constexpr int nPointsPerInch = 72; |
| if (nWidth != 0 && nHeight != 0) { |
| metadata->horizontal_dpi = nPixelWidth / nWidth * nPointsPerInch; |
| metadata->vertical_dpi = nPixelHeight / nHeight * nPointsPerInch; |
| } |
| |
| metadata->bits_per_pixel = 0; |
| metadata->colorspace = FPDF_COLORSPACE_UNKNOWN; |
| |
| CPDF_Page* pPage = CPDFPageFromFPDFPage(page); |
| if (!pPage || !pPage->m_pDocument.Get() || !pImg->GetStream()) |
| return true; |
| |
| auto pSource = pdfium::MakeRetain<CPDF_DIBSource>(); |
| if (!pSource->StartLoadDIBSource(pPage->m_pDocument.Get(), pImg->GetStream(), |
| false, nullptr, |
| pPage->m_pPageResources.Get())) { |
| return true; |
| } |
| |
| metadata->bits_per_pixel = pSource->GetBPP(); |
| if (pSource->GetColorSpace()) |
| metadata->colorspace = pSource->GetColorSpace()->GetFamily(); |
| |
| return true; |
| } |