// 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 <algorithm> #include "core/fpdfapi/fpdf_parser/include/cpdf_array.h" #include "core/fpdfapi/fpdf_parser/include/cpdf_document.h" #include "core/fpdfapi/fpdf_parser/include/cpdf_number.h" #include "core/fpdfapi/fpdf_parser/include/cpdf_stream.h" #include "core/fxcrt/include/fx_ext.h" #include "fpdfsdk/include/fsdk_baseannot.h" #include "fpdfsdk/include/fsdk_define.h" #include "fpdfsdk/include/fsdk_mgr.h" #ifdef PDF_ENABLE_XFA #include "fpdfsdk/fpdfxfa/include/fpdfxfa_doc.h" #endif // PDF_ENABLE_XFA namespace { const float kMinWidth = 1.0f; const float kMinHeight = 1.0f; int gAfxGetTimeZoneInSeconds(int8_t tzhour, uint8_t tzminute) { return (int)tzhour * 3600 + (int)tzminute * (tzhour >= 0 ? 60 : -60); } bool gAfxIsLeapYear(int16_t year) { return ((year % 400 == 0) || ((year % 4 == 0) && (year % 100 != 0))); } uint16_t gAfxGetYearDays(int16_t year) { return (gAfxIsLeapYear(year) ? 366 : 365); } uint8_t gAfxGetMonthDays(int16_t year, uint8_t month) { uint8_t mDays; switch (month) { case 1: case 3: case 5: case 7: case 8: case 10: case 12: mDays = 31; break; case 4: case 6: case 9: case 11: mDays = 30; break; case 2: if (gAfxIsLeapYear(year)) mDays = 29; else mDays = 28; break; default: mDays = 0; break; } return mDays; } } // namespace CPDFSDK_DateTime::CPDFSDK_DateTime() { ResetDateTime(); } CPDFSDK_DateTime::CPDFSDK_DateTime(const CFX_ByteString& dtStr) { ResetDateTime(); FromPDFDateTimeString(dtStr); } CPDFSDK_DateTime::CPDFSDK_DateTime(const CPDFSDK_DateTime& datetime) { operator=(datetime); } CPDFSDK_DateTime::CPDFSDK_DateTime(const FX_SYSTEMTIME& st) { operator=(st); } void CPDFSDK_DateTime::ResetDateTime() { tzset(); time_t curTime; time(&curTime); struct tm* newtime = localtime(&curTime); dt.year = newtime->tm_year + 1900; dt.month = newtime->tm_mon + 1; dt.day = newtime->tm_mday; dt.hour = newtime->tm_hour; dt.minute = newtime->tm_min; dt.second = newtime->tm_sec; } CPDFSDK_DateTime& CPDFSDK_DateTime::operator=( const CPDFSDK_DateTime& datetime) { FXSYS_memcpy(&dt, &datetime.dt, sizeof(FX_DATETIME)); return *this; } CPDFSDK_DateTime& CPDFSDK_DateTime::operator=(const FX_SYSTEMTIME& st) { tzset(); dt.year = static_cast<int16_t>(st.wYear); dt.month = static_cast<uint8_t>(st.wMonth); dt.day = static_cast<uint8_t>(st.wDay); dt.hour = static_cast<uint8_t>(st.wHour); dt.minute = static_cast<uint8_t>(st.wMinute); dt.second = static_cast<uint8_t>(st.wSecond); return *this; } bool CPDFSDK_DateTime::operator==(const CPDFSDK_DateTime& datetime) const { return (FXSYS_memcmp(&dt, &datetime.dt, sizeof(FX_DATETIME)) == 0); } bool CPDFSDK_DateTime::operator!=(const CPDFSDK_DateTime& datetime) const { return !(*this == datetime); } time_t CPDFSDK_DateTime::ToTime_t() const { struct tm newtime; newtime.tm_year = dt.year - 1900; newtime.tm_mon = dt.month - 1; newtime.tm_mday = dt.day; newtime.tm_hour = dt.hour; newtime.tm_min = dt.minute; newtime.tm_sec = dt.second; return mktime(&newtime); } CPDFSDK_DateTime& CPDFSDK_DateTime::FromPDFDateTimeString( const CFX_ByteString& dtStr) { int strLength = dtStr.GetLength(); if (strLength <= 0) return *this; int i = 0; while (i < strLength && !std::isdigit(dtStr[i])) ++i; if (i >= strLength) return *this; int j = 0; int k = 0; FX_CHAR ch; while (i < strLength && j < 4) { ch = dtStr[i]; k = k * 10 + FXSYS_toDecimalDigit(ch); j++; if (!std::isdigit(ch)) break; i++; } dt.year = static_cast<int16_t>(k); if (i >= strLength || j < 4) return *this; j = 0; k = 0; while (i < strLength && j < 2) { ch = dtStr[i]; k = k * 10 + FXSYS_toDecimalDigit(ch); j++; if (!std::isdigit(ch)) break; i++; } dt.month = static_cast<uint8_t>(k); if (i >= strLength || j < 2) return *this; j = 0; k = 0; while (i < strLength && j < 2) { ch = dtStr[i]; k = k * 10 + FXSYS_toDecimalDigit(ch); j++; if (!std::isdigit(ch)) break; i++; } dt.day = static_cast<uint8_t>(k); if (i >= strLength || j < 2) return *this; j = 0; k = 0; while (i < strLength && j < 2) { ch = dtStr[i]; k = k * 10 + FXSYS_toDecimalDigit(ch); j++; if (!std::isdigit(ch)) break; i++; } dt.hour = static_cast<uint8_t>(k); if (i >= strLength || j < 2) return *this; j = 0; k = 0; while (i < strLength && j < 2) { ch = dtStr[i]; k = k * 10 + FXSYS_toDecimalDigit(ch); j++; if (!std::isdigit(ch)) break; i++; } dt.minute = static_cast<uint8_t>(k); if (i >= strLength || j < 2) return *this; j = 0; k = 0; while (i < strLength && j < 2) { ch = dtStr[i]; k = k * 10 + FXSYS_toDecimalDigit(ch); j++; if (!std::isdigit(ch)) break; i++; } dt.second = static_cast<uint8_t>(k); if (i >= strLength || j < 2) return *this; ch = dtStr[i++]; if (ch != '-' && ch != '+') return *this; if (ch == '-') dt.tzHour = -1; else dt.tzHour = 1; j = 0; k = 0; while (i < strLength && j < 2) { ch = dtStr[i]; k = k * 10 + FXSYS_toDecimalDigit(ch); j++; if (!std::isdigit(ch)) break; i++; } dt.tzHour *= static_cast<int8_t>(k); if (i >= strLength || j < 2) return *this; if (dtStr[i++] != '\'') return *this; j = 0; k = 0; while (i < strLength && j < 2) { ch = dtStr[i]; k = k * 10 + FXSYS_toDecimalDigit(ch); j++; if (!std::isdigit(ch)) break; i++; } dt.tzMinute = static_cast<uint8_t>(k); return *this; } CFX_ByteString CPDFSDK_DateTime::ToCommonDateTimeString() { CFX_ByteString str1; str1.Format("%04d-%02u-%02u %02u:%02u:%02u ", dt.year, dt.month, dt.day, dt.hour, dt.minute, dt.second); if (dt.tzHour < 0) str1 += "-"; else str1 += "+"; CFX_ByteString str2; str2.Format("%02d:%02u", std::abs(static_cast<int>(dt.tzHour)), dt.tzMinute); return str1 + str2; } CFX_ByteString CPDFSDK_DateTime::ToPDFDateTimeString() { CFX_ByteString dtStr; char tempStr[32]; memset(tempStr, 0, sizeof(tempStr)); FXSYS_snprintf(tempStr, sizeof(tempStr) - 1, "D:%04d%02u%02u%02u%02u%02u", dt.year, dt.month, dt.day, dt.hour, dt.minute, dt.second); dtStr = CFX_ByteString(tempStr); if (dt.tzHour < 0) dtStr += CFX_ByteString("-"); else dtStr += CFX_ByteString("+"); memset(tempStr, 0, sizeof(tempStr)); FXSYS_snprintf(tempStr, sizeof(tempStr) - 1, "%02d'%02u'", std::abs(static_cast<int>(dt.tzHour)), dt.tzMinute); dtStr += CFX_ByteString(tempStr); return dtStr; } void CPDFSDK_DateTime::ToSystemTime(FX_SYSTEMTIME& st) { time_t t = this->ToTime_t(); struct tm* pTime = localtime(&t); if (pTime) { st.wYear = static_cast<uint16_t>(pTime->tm_year) + 1900; st.wMonth = static_cast<uint16_t>(pTime->tm_mon) + 1; st.wDay = static_cast<uint16_t>(pTime->tm_mday); st.wDayOfWeek = static_cast<uint16_t>(pTime->tm_wday); st.wHour = static_cast<uint16_t>(pTime->tm_hour); st.wMinute = static_cast<uint16_t>(pTime->tm_min); st.wSecond = static_cast<uint16_t>(pTime->tm_sec); st.wMilliseconds = 0; } } CPDFSDK_DateTime CPDFSDK_DateTime::ToGMT() const { CPDFSDK_DateTime new_dt = *this; new_dt.AddSeconds( -gAfxGetTimeZoneInSeconds(new_dt.dt.tzHour, new_dt.dt.tzMinute)); new_dt.dt.tzHour = 0; new_dt.dt.tzMinute = 0; return new_dt; } CPDFSDK_DateTime& CPDFSDK_DateTime::AddDays(short days) { if (days == 0) return *this; int16_t y = dt.year; uint8_t m = dt.month; uint8_t d = dt.day; int ldays = days; if (ldays > 0) { int16_t yy = y; if ((static_cast<uint16_t>(m) * 100 + d) > 300) yy++; int ydays = gAfxGetYearDays(yy); int mdays; while (ldays >= ydays) { y++; ldays -= ydays; yy++; mdays = gAfxGetMonthDays(y, m); if (d > mdays) { m++; d -= mdays; } ydays = gAfxGetYearDays(yy); } mdays = gAfxGetMonthDays(y, m) - d + 1; while (ldays >= mdays) { ldays -= mdays; m++; d = 1; mdays = gAfxGetMonthDays(y, m); } d += ldays; } else { ldays *= -1; int16_t yy = y; if ((static_cast<uint16_t>(m) * 100 + d) < 300) yy--; int ydays = gAfxGetYearDays(yy); while (ldays >= ydays) { y--; ldays -= ydays; yy--; int mdays = gAfxGetMonthDays(y, m); if (d > mdays) { m++; d -= mdays; } ydays = gAfxGetYearDays(yy); } while (ldays >= d) { ldays -= d; m--; d = gAfxGetMonthDays(y, m); } d -= ldays; } dt.year = y; dt.month = m; dt.day = d; return *this; } CPDFSDK_DateTime& CPDFSDK_DateTime::AddSeconds(int seconds) { if (seconds == 0) return *this; int n; int days; n = dt.hour * 3600 + dt.minute * 60 + dt.second + seconds; if (n < 0) { days = (n - 86399) / 86400; n -= days * 86400; } else { days = n / 86400; n %= 86400; } dt.hour = static_cast<uint8_t>(n / 3600); dt.hour %= 24; n %= 3600; dt.minute = static_cast<uint8_t>(n / 60); dt.second = static_cast<uint8_t>(n % 60); if (days != 0) AddDays(days); return *this; } CPDFSDK_Annot::CPDFSDK_Annot(CPDFSDK_PageView* pPageView) : m_pPageView(pPageView), m_bSelected(FALSE), m_nTabOrder(-1) {} CPDFSDK_Annot::~CPDFSDK_Annot() {} #ifdef PDF_ENABLE_XFA FX_BOOL CPDFSDK_Annot::IsXFAField() { return FALSE; } CXFA_FFWidget* CPDFSDK_Annot::GetXFAWidget() const { return nullptr; } #endif // PDF_ENABLE_XFA FX_FLOAT CPDFSDK_Annot::GetMinWidth() const { return kMinWidth; } FX_FLOAT CPDFSDK_Annot::GetMinHeight() const { return kMinHeight; } int CPDFSDK_Annot::GetLayoutOrder() const { return 5; } CPDF_Annot* CPDFSDK_Annot::GetPDFAnnot() const { return nullptr; } CFX_ByteString CPDFSDK_Annot::GetType() const { return ""; } CFX_ByteString CPDFSDK_Annot::GetSubType() const { return ""; } void CPDFSDK_Annot::SetRect(const CFX_FloatRect& rect) {} CFX_FloatRect CPDFSDK_Annot::GetRect() const { return CFX_FloatRect(); } void CPDFSDK_Annot::Annot_OnDraw(CFX_RenderDevice* pDevice, CFX_Matrix* pUser2Device, CPDF_RenderOptions* pOptions) {} CPDFSDK_BAAnnot::CPDFSDK_BAAnnot(CPDF_Annot* pAnnot, CPDFSDK_PageView* pPageView) : CPDFSDK_Annot(pPageView), m_pAnnot(pAnnot) {} CPDFSDK_BAAnnot::~CPDFSDK_BAAnnot() {} CPDF_Annot* CPDFSDK_BAAnnot::GetPDFAnnot() const { return m_pAnnot; } FX_BOOL CPDFSDK_Annot::IsSelected() { return m_bSelected; } void CPDFSDK_Annot::SetSelected(FX_BOOL bSelected) { m_bSelected = bSelected; } int CPDFSDK_Annot::GetTabOrder() { return m_nTabOrder; } void CPDFSDK_Annot::SetTabOrder(int iTabOrder) { m_nTabOrder = iTabOrder; } CPDF_Dictionary* CPDFSDK_BAAnnot::GetAnnotDict() const { return m_pAnnot->GetAnnotDict(); } void CPDFSDK_BAAnnot::SetRect(const CFX_FloatRect& rect) { ASSERT(rect.right - rect.left >= GetMinWidth()); ASSERT(rect.top - rect.bottom >= GetMinHeight()); m_pAnnot->GetAnnotDict()->SetAtRect("Rect", rect); } CFX_FloatRect CPDFSDK_BAAnnot::GetRect() const { CFX_FloatRect rect; m_pAnnot->GetRect(rect); return rect; } CFX_ByteString CPDFSDK_BAAnnot::GetType() const { return m_pAnnot->GetSubType(); } CFX_ByteString CPDFSDK_BAAnnot::GetSubType() const { return ""; } void CPDFSDK_BAAnnot::DrawAppearance(CFX_RenderDevice* pDevice, const CFX_Matrix* pUser2Device, CPDF_Annot::AppearanceMode mode, const CPDF_RenderOptions* pOptions) { m_pAnnot->DrawAppearance(m_pPageView->GetPDFPage(), pDevice, pUser2Device, mode, pOptions); } FX_BOOL CPDFSDK_BAAnnot::IsAppearanceValid() { return !!m_pAnnot->GetAnnotDict()->GetDictBy("AP"); } FX_BOOL CPDFSDK_BAAnnot::IsAppearanceValid(CPDF_Annot::AppearanceMode mode) { CPDF_Dictionary* pAP = m_pAnnot->GetAnnotDict()->GetDictBy("AP"); if (!pAP) return FALSE; // Choose the right sub-ap const FX_CHAR* ap_entry = "N"; if (mode == CPDF_Annot::Down) ap_entry = "D"; else if (mode == CPDF_Annot::Rollover) ap_entry = "R"; if (!pAP->KeyExist(ap_entry)) ap_entry = "N"; // Get the AP stream or subdirectory CPDF_Object* psub = pAP->GetDirectObjectBy(ap_entry); return !!psub; } void CPDFSDK_BAAnnot::DrawBorder(CFX_RenderDevice* pDevice, const CFX_Matrix* pUser2Device, const CPDF_RenderOptions* pOptions) { m_pAnnot->DrawBorder(pDevice, pUser2Device, pOptions); } void CPDFSDK_BAAnnot::ClearCachedAP() { m_pAnnot->ClearCachedAP(); } void CPDFSDK_BAAnnot::SetContents(const CFX_WideString& sContents) { if (sContents.IsEmpty()) m_pAnnot->GetAnnotDict()->RemoveAt("Contents"); else m_pAnnot->GetAnnotDict()->SetAtString("Contents", PDF_EncodeText(sContents)); } CFX_WideString CPDFSDK_BAAnnot::GetContents() const { return m_pAnnot->GetAnnotDict()->GetUnicodeTextBy("Contents"); } void CPDFSDK_BAAnnot::SetAnnotName(const CFX_WideString& sName) { if (sName.IsEmpty()) m_pAnnot->GetAnnotDict()->RemoveAt("NM"); else m_pAnnot->GetAnnotDict()->SetAtString("NM", PDF_EncodeText(sName)); } CFX_WideString CPDFSDK_BAAnnot::GetAnnotName() const { return m_pAnnot->GetAnnotDict()->GetUnicodeTextBy("NM"); } void CPDFSDK_BAAnnot::SetModifiedDate(const FX_SYSTEMTIME& st) { CPDFSDK_DateTime dt(st); CFX_ByteString str = dt.ToPDFDateTimeString(); if (str.IsEmpty()) m_pAnnot->GetAnnotDict()->RemoveAt("M"); else m_pAnnot->GetAnnotDict()->SetAtString("M", str); } FX_SYSTEMTIME CPDFSDK_BAAnnot::GetModifiedDate() const { FX_SYSTEMTIME systime; CFX_ByteString str = m_pAnnot->GetAnnotDict()->GetStringBy("M"); CPDFSDK_DateTime dt(str); dt.ToSystemTime(systime); return systime; } void CPDFSDK_BAAnnot::SetFlags(uint32_t nFlags) { m_pAnnot->GetAnnotDict()->SetAtInteger("F", nFlags); } uint32_t CPDFSDK_BAAnnot::GetFlags() const { return m_pAnnot->GetAnnotDict()->GetIntegerBy("F"); } void CPDFSDK_BAAnnot::SetAppState(const CFX_ByteString& str) { if (str.IsEmpty()) m_pAnnot->GetAnnotDict()->RemoveAt("AS"); else m_pAnnot->GetAnnotDict()->SetAtString("AS", str); } CFX_ByteString CPDFSDK_BAAnnot::GetAppState() const { return m_pAnnot->GetAnnotDict()->GetStringBy("AS"); } void CPDFSDK_BAAnnot::SetStructParent(int key) { m_pAnnot->GetAnnotDict()->SetAtInteger("StructParent", key); } int CPDFSDK_BAAnnot::GetStructParent() const { return m_pAnnot->GetAnnotDict()->GetIntegerBy("StructParent"); } // border void CPDFSDK_BAAnnot::SetBorderWidth(int nWidth) { CPDF_Array* pBorder = m_pAnnot->GetAnnotDict()->GetArrayBy("Border"); if (pBorder) { pBorder->SetAt(2, new CPDF_Number(nWidth)); } else { CPDF_Dictionary* pBSDict = m_pAnnot->GetAnnotDict()->GetDictBy("BS"); if (!pBSDict) { pBSDict = new CPDF_Dictionary; m_pAnnot->GetAnnotDict()->SetAt("BS", pBSDict); } pBSDict->SetAtInteger("W", nWidth); } } int CPDFSDK_BAAnnot::GetBorderWidth() const { if (CPDF_Array* pBorder = m_pAnnot->GetAnnotDict()->GetArrayBy("Border")) { return pBorder->GetIntegerAt(2); } if (CPDF_Dictionary* pBSDict = m_pAnnot->GetAnnotDict()->GetDictBy("BS")) { return pBSDict->GetIntegerBy("W", 1); } return 1; } void CPDFSDK_BAAnnot::SetBorderStyle(BorderStyle nStyle) { CPDF_Dictionary* pBSDict = m_pAnnot->GetAnnotDict()->GetDictBy("BS"); if (!pBSDict) { pBSDict = new CPDF_Dictionary; m_pAnnot->GetAnnotDict()->SetAt("BS", pBSDict); } switch (nStyle) { case BorderStyle::SOLID: pBSDict->SetAtName("S", "S"); break; case BorderStyle::DASH: pBSDict->SetAtName("S", "D"); break; case BorderStyle::BEVELED: pBSDict->SetAtName("S", "B"); break; case BorderStyle::INSET: pBSDict->SetAtName("S", "I"); break; case BorderStyle::UNDERLINE: pBSDict->SetAtName("S", "U"); break; default: break; } } BorderStyle CPDFSDK_BAAnnot::GetBorderStyle() const { CPDF_Dictionary* pBSDict = m_pAnnot->GetAnnotDict()->GetDictBy("BS"); if (pBSDict) { CFX_ByteString sBorderStyle = pBSDict->GetStringBy("S", "S"); if (sBorderStyle == "S") return BorderStyle::SOLID; if (sBorderStyle == "D") return BorderStyle::DASH; if (sBorderStyle == "B") return BorderStyle::BEVELED; if (sBorderStyle == "I") return BorderStyle::INSET; if (sBorderStyle == "U") return BorderStyle::UNDERLINE; } CPDF_Array* pBorder = m_pAnnot->GetAnnotDict()->GetArrayBy("Border"); if (pBorder) { if (pBorder->GetCount() >= 4) { CPDF_Array* pDP = pBorder->GetArrayAt(3); if (pDP && pDP->GetCount() > 0) return BorderStyle::DASH; } } return BorderStyle::SOLID; } void CPDFSDK_BAAnnot::SetColor(FX_COLORREF color) { CPDF_Array* pArray = new CPDF_Array; pArray->AddNumber((FX_FLOAT)FXSYS_GetRValue(color) / 255.0f); pArray->AddNumber((FX_FLOAT)FXSYS_GetGValue(color) / 255.0f); pArray->AddNumber((FX_FLOAT)FXSYS_GetBValue(color) / 255.0f); m_pAnnot->GetAnnotDict()->SetAt("C", pArray); } void CPDFSDK_BAAnnot::RemoveColor() { m_pAnnot->GetAnnotDict()->RemoveAt("C"); } FX_BOOL CPDFSDK_BAAnnot::GetColor(FX_COLORREF& color) const { if (CPDF_Array* pEntry = m_pAnnot->GetAnnotDict()->GetArrayBy("C")) { size_t nCount = pEntry->GetCount(); if (nCount == 1) { FX_FLOAT g = pEntry->GetNumberAt(0) * 255; color = FXSYS_RGB((int)g, (int)g, (int)g); return TRUE; } else if (nCount == 3) { FX_FLOAT r = pEntry->GetNumberAt(0) * 255; FX_FLOAT g = pEntry->GetNumberAt(1) * 255; FX_FLOAT b = pEntry->GetNumberAt(2) * 255; color = FXSYS_RGB((int)r, (int)g, (int)b); return TRUE; } else if (nCount == 4) { FX_FLOAT c = pEntry->GetNumberAt(0); FX_FLOAT m = pEntry->GetNumberAt(1); FX_FLOAT y = pEntry->GetNumberAt(2); FX_FLOAT k = pEntry->GetNumberAt(3); FX_FLOAT r = 1.0f - std::min(1.0f, c + k); FX_FLOAT g = 1.0f - std::min(1.0f, m + k); FX_FLOAT b = 1.0f - std::min(1.0f, y + k); color = FXSYS_RGB((int)(r * 255), (int)(g * 255), (int)(b * 255)); return TRUE; } } return FALSE; } void CPDFSDK_BAAnnot::WriteAppearance(const CFX_ByteString& sAPType, const CFX_FloatRect& rcBBox, const CFX_Matrix& matrix, const CFX_ByteString& sContents, const CFX_ByteString& sAPState) { CPDF_Dictionary* pAPDict = m_pAnnot->GetAnnotDict()->GetDictBy("AP"); if (!pAPDict) { pAPDict = new CPDF_Dictionary; m_pAnnot->GetAnnotDict()->SetAt("AP", pAPDict); } CPDF_Stream* pStream = nullptr; CPDF_Dictionary* pParentDict = nullptr; if (sAPState.IsEmpty()) { pParentDict = pAPDict; pStream = pAPDict->GetStreamBy(sAPType); } else { CPDF_Dictionary* pAPTypeDict = pAPDict->GetDictBy(sAPType); if (!pAPTypeDict) { pAPTypeDict = new CPDF_Dictionary; pAPDict->SetAt(sAPType, pAPTypeDict); } pParentDict = pAPTypeDict; pStream = pAPTypeDict->GetStreamBy(sAPState); } if (!pStream) { pStream = new CPDF_Stream(nullptr, 0, nullptr); CPDF_Document* pDoc = m_pPageView->GetPDFDocument(); int32_t objnum = pDoc->AddIndirectObject(pStream); pParentDict->SetAtReference(sAPType, pDoc, objnum); } CPDF_Dictionary* pStreamDict = pStream->GetDict(); if (!pStreamDict) { pStreamDict = new CPDF_Dictionary; pStreamDict->SetAtName("Type", "XObject"); pStreamDict->SetAtName("Subtype", "Form"); pStreamDict->SetAtInteger("FormType", 1); pStream->InitStream(nullptr, 0, pStreamDict); } if (pStreamDict) { pStreamDict->SetAtMatrix("Matrix", matrix); pStreamDict->SetAtRect("BBox", rcBBox); } pStream->SetData((uint8_t*)sContents.c_str(), sContents.GetLength(), FALSE, FALSE); } FX_BOOL CPDFSDK_BAAnnot::CreateFormFiller() { return TRUE; } FX_BOOL CPDFSDK_BAAnnot::IsVisible() const { uint32_t nFlags = GetFlags(); return !((nFlags & ANNOTFLAG_INVISIBLE) || (nFlags & ANNOTFLAG_HIDDEN) || (nFlags & ANNOTFLAG_NOVIEW)); } CPDF_Action CPDFSDK_BAAnnot::GetAction() const { return CPDF_Action(m_pAnnot->GetAnnotDict()->GetDictBy("A")); } void CPDFSDK_BAAnnot::SetAction(const CPDF_Action& action) { ASSERT(action.GetDict()); if (action.GetDict() != m_pAnnot->GetAnnotDict()->GetDictBy("A")) { CPDF_Document* pDoc = m_pPageView->GetPDFDocument(); CPDF_Dictionary* pDict = action.GetDict(); if (pDict && pDict->GetObjNum() == 0) { pDoc->AddIndirectObject(pDict); } m_pAnnot->GetAnnotDict()->SetAtReference("A", pDoc, pDict->GetObjNum()); } } void CPDFSDK_BAAnnot::RemoveAction() { m_pAnnot->GetAnnotDict()->RemoveAt("A"); } CPDF_AAction CPDFSDK_BAAnnot::GetAAction() const { return CPDF_AAction(m_pAnnot->GetAnnotDict()->GetDictBy("AA")); } void CPDFSDK_BAAnnot::SetAAction(const CPDF_AAction& aa) { if (aa.GetDict() != m_pAnnot->GetAnnotDict()->GetDictBy("AA")) m_pAnnot->GetAnnotDict()->SetAt("AA", aa.GetDict()); } void CPDFSDK_BAAnnot::RemoveAAction() { m_pAnnot->GetAnnotDict()->RemoveAt("AA"); } CPDF_Action CPDFSDK_BAAnnot::GetAAction(CPDF_AAction::AActionType eAAT) { CPDF_AAction AAction = GetAAction(); if (AAction.ActionExist(eAAT)) return AAction.GetAction(eAAT); if (eAAT == CPDF_AAction::ButtonUp) return GetAction(); return CPDF_Action(); } void CPDFSDK_BAAnnot::Annot_OnDraw(CFX_RenderDevice* pDevice, CFX_Matrix* pUser2Device, CPDF_RenderOptions* pOptions) { m_pAnnot->GetAPForm(m_pPageView->GetPDFPage(), CPDF_Annot::Normal); m_pAnnot->DrawAppearance(m_pPageView->GetPDFPage(), pDevice, pUser2Device, CPDF_Annot::Normal, nullptr); } UnderlyingPageType* CPDFSDK_Annot::GetUnderlyingPage() { #ifdef PDF_ENABLE_XFA return GetPDFXFAPage(); #else // PDF_ENABLE_XFA return GetPDFPage(); #endif // PDF_ENABLE_XFA } CPDF_Page* CPDFSDK_Annot::GetPDFPage() { return m_pPageView ? m_pPageView->GetPDFPage() : nullptr; } #ifdef PDF_ENABLE_XFA CPDFXFA_Page* CPDFSDK_Annot::GetPDFXFAPage() { return m_pPageView ? m_pPageView->GetPDFXFAPage() : nullptr; } #endif // PDF_ENABLE_XFA