// 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_pageobject.h" #include "fpdfsdk/fsdk_define.h" #include "third_party/base/ptr_util.h" namespace { bool LoadJpegHelper(FPDF_PAGE* pages, int nCount, FPDF_PAGEOBJECT image_object, FPDF_FILEACCESS* fileAccess, bool inlineJpeg) { if (!image_object || !fileAccess) return false; CFX_RetainPtr pFile = MakeSeekableReadStream(fileAccess); CPDF_ImageObject* pImgObj = static_cast(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 DLLEXPORT FPDF_PAGEOBJECT STDCALL FPDFPageObj_NewImageObj(FPDF_DOCUMENT document) { CPDF_Document* pDoc = CPDFDocumentFromFPDFDocument(document); if (!pDoc) return nullptr; auto pImageObj = pdfium::MakeUnique(); pImageObj->SetImage(pdfium::MakeRetain(pDoc)); return pImageObj.release(); } DLLEXPORT FPDF_BOOL STDCALL FPDFImageObj_LoadJpegFile(FPDF_PAGE* pages, int nCount, FPDF_PAGEOBJECT image_object, FPDF_FILEACCESS* fileAccess) { return LoadJpegHelper(pages, nCount, image_object, fileAccess, false); } DLLEXPORT FPDF_BOOL STDCALL FPDFImageObj_LoadJpegFileInline(FPDF_PAGE* pages, int nCount, FPDF_PAGEOBJECT image_object, FPDF_FILEACCESS* fileAccess) { return LoadJpegHelper(pages, nCount, image_object, fileAccess, true); } DLLEXPORT FPDF_BOOL STDCALL 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(image_object); pImgObj->set_matrix(CFX_Matrix(static_cast(a), static_cast(b), static_cast(c), static_cast(d), static_cast(e), static_cast(f))); pImgObj->CalcBoundingBox(); pImgObj->SetDirty(true); return true; } DLLEXPORT FPDF_BOOL STDCALL 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(image_object); for (int index = 0; index < nCount; index++) { CPDF_Page* pPage = CPDFPageFromFPDFPage(pages[index]); if (pPage) pImgObj->GetImage()->ResetCache(pPage, nullptr); } CFX_RetainPtr holder(CFXBitmapFromFPDFBitmap(bitmap)); pImgObj->GetImage()->SetImage(holder); pImgObj->CalcBoundingBox(); pImgObj->SetDirty(true); return true; } DLLEXPORT FPDF_BITMAP STDCALL FPDFImageObj_GetBitmap(FPDF_PAGEOBJECT image_object) { CPDF_PageObject* pObj = CPDFPageObjectFromFPDFPageObject(image_object); if (!pObj || !pObj->IsImage()) return nullptr; CFX_RetainPtr pImg = pObj->AsImage()->GetImage(); if (!pImg) return nullptr; CFX_RetainPtr pSource = pImg->LoadDIBSource(); if (!pSource) return nullptr; CFX_RetainPtr 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(); } DLLEXPORT unsigned long STDCALL FPDFImageObj_GetImageDataDecoded(FPDF_PAGEOBJECT image_object, void* buffer, unsigned long buflen) { CPDF_PageObject* pObj = CPDFPageObjectFromFPDFPageObject(image_object); if (!pObj || !pObj->IsImage()) return 0; CFX_RetainPtr pImg = pObj->AsImage()->GetImage(); if (!pImg) return 0; CPDF_Stream* pImgStream = pImg->GetStream(); if (!pImgStream) return 0; return DecodeStreamMaybeCopyAndReturnLength(pImgStream, buffer, buflen); } DLLEXPORT unsigned long STDCALL FPDFImageObj_GetImageDataRaw(FPDF_PAGEOBJECT image_object, void* buffer, unsigned long buflen) { CPDF_PageObject* pObj = CPDFPageObjectFromFPDFPageObject(image_object); if (!pObj || !pObj->IsImage()) return 0; CFX_RetainPtr 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; }