| // 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 = (int16_t)st.wYear; |
| dt.month = (uint8_t)st.wMonth; |
| dt.day = (uint8_t)st.wDay; |
| dt.hour = (uint8_t)st.wHour; |
| dt.minute = (uint8_t)st.wMinute; |
| dt.second = (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); |
| } |
| |
| bool CPDFSDK_DateTime::operator>(const CPDFSDK_DateTime& datetime) const { |
| CPDFSDK_DateTime dt1 = ToGMT(); |
| CPDFSDK_DateTime dt2 = datetime.ToGMT(); |
| int d1 = |
| (((int)dt1.dt.year) << 16) | (((int)dt1.dt.month) << 8) | (int)dt1.dt.day; |
| int d2 = (((int)dt1.dt.hour) << 16) | (((int)dt1.dt.minute) << 8) | |
| (int)dt1.dt.second; |
| int d3 = |
| (((int)dt2.dt.year) << 16) | (((int)dt2.dt.month) << 8) | (int)dt2.dt.day; |
| int d4 = (((int)dt2.dt.hour) << 16) | (((int)dt2.dt.minute) << 8) | |
| (int)dt2.dt.second; |
| |
| return d1 > d3 || d2 > d4; |
| } |
| |
| bool CPDFSDK_DateTime::operator>=(const CPDFSDK_DateTime& datetime) const { |
| CPDFSDK_DateTime dt1 = ToGMT(); |
| CPDFSDK_DateTime dt2 = datetime.ToGMT(); |
| int d1 = |
| (((int)dt1.dt.year) << 16) | (((int)dt1.dt.month) << 8) | (int)dt1.dt.day; |
| int d2 = (((int)dt1.dt.hour) << 16) | (((int)dt1.dt.minute) << 8) | |
| (int)dt1.dt.second; |
| int d3 = |
| (((int)dt2.dt.year) << 16) | (((int)dt2.dt.month) << 8) | (int)dt2.dt.day; |
| int d4 = (((int)dt2.dt.hour) << 16) | (((int)dt2.dt.minute) << 8) | |
| (int)dt2.dt.second; |
| |
| return d1 >= d3 || d2 >= d4; |
| } |
| |
| bool CPDFSDK_DateTime::operator<(const CPDFSDK_DateTime& datetime) const { |
| CPDFSDK_DateTime dt1 = ToGMT(); |
| CPDFSDK_DateTime dt2 = datetime.ToGMT(); |
| int d1 = |
| (((int)dt1.dt.year) << 16) | (((int)dt1.dt.month) << 8) | (int)dt1.dt.day; |
| int d2 = (((int)dt1.dt.hour) << 16) | (((int)dt1.dt.minute) << 8) | |
| (int)dt1.dt.second; |
| int d3 = |
| (((int)dt2.dt.year) << 16) | (((int)dt2.dt.month) << 8) | (int)dt2.dt.day; |
| int d4 = (((int)dt2.dt.hour) << 16) | (((int)dt2.dt.minute) << 8) | |
| (int)dt2.dt.second; |
| |
| return d1 < d3 || d2 < d4; |
| } |
| |
| bool CPDFSDK_DateTime::operator<=(const CPDFSDK_DateTime& datetime) const { |
| CPDFSDK_DateTime dt1 = ToGMT(); |
| CPDFSDK_DateTime dt2 = datetime.ToGMT(); |
| int d1 = |
| (((int)dt1.dt.year) << 16) | (((int)dt1.dt.month) << 8) | (int)dt1.dt.day; |
| int d2 = (((int)dt1.dt.hour) << 16) | (((int)dt1.dt.minute) << 8) | |
| (int)dt1.dt.second; |
| int d3 = |
| (((int)dt2.dt.year) << 16) | (((int)dt2.dt.month) << 8) | (int)dt2.dt.day; |
| int d4 = (((int)dt2.dt.hour) << 16) | (((int)dt2.dt.minute) << 8) | |
| (int)dt2.dt.second; |
| |
| return d1 <= d3 || d2 <= d4; |
| } |
| |
| CPDFSDK_DateTime::operator time_t() { |
| 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) { |
| int i = 0; |
| int j, k; |
| FX_CHAR ch; |
| while (i < strLength && !std::isdigit(dtStr[i])) |
| ++i; |
| |
| if (i >= strLength) |
| return *this; |
| |
| j = 0; |
| k = 0; |
| while (i < strLength && j < 4) { |
| ch = dtStr[i]; |
| k = k * 10 + FXSYS_toDecimalDigit(ch); |
| j++; |
| if (!std::isdigit(ch)) |
| break; |
| i++; |
| } |
| dt.year = (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 = (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 = (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 = (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 = (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 = (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 *= (FX_CHAR)k; |
| if (i >= strLength || j < 2) |
| return *this; |
| |
| ch = dtStr[i++]; |
| if (ch != '\'') |
| 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 = (uint8_t)k; |
| if (i >= strLength || j < 2) |
| return *this; |
| } |
| |
| return *this; |
| } |
| |
| CFX_ByteString CPDFSDK_DateTime::ToCommonDateTimeString() { |
| CFX_ByteString str1; |
| str1.Format("%04d-%02d-%02d %02d:%02d:%02d ", 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:%02d", abs(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%02d%02d%02d%02d%02d", |
| 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'%02d'", abs(dt.tzHour), |
| dt.tzMinute); |
| dtStr += CFX_ByteString(tempStr); |
| return dtStr; |
| } |
| |
| void CPDFSDK_DateTime::ToSystemTime(FX_SYSTEMTIME& st) { |
| CPDFSDK_DateTime dt = *this; |
| time_t t = (time_t)dt; |
| struct tm* pTime = localtime(&t); |
| if (pTime) { |
| st.wYear = (uint16_t)pTime->tm_year + 1900; |
| st.wMonth = (uint16_t)pTime->tm_mon + 1; |
| st.wDay = (uint16_t)pTime->tm_mday; |
| st.wDayOfWeek = (uint16_t)pTime->tm_wday; |
| st.wHour = (uint16_t)pTime->tm_hour; |
| st.wMinute = (uint16_t)pTime->tm_min; |
| st.wSecond = (uint16_t)pTime->tm_sec; |
| st.wMilliseconds = 0; |
| } |
| } |
| |
| CPDFSDK_DateTime CPDFSDK_DateTime::ToGMT() const { |
| CPDFSDK_DateTime dt = *this; |
| dt.AddSeconds(-gAfxGetTimeZoneInSeconds(dt.dt.tzHour, dt.dt.tzMinute)); |
| dt.dt.tzHour = 0; |
| dt.dt.tzMinute = 0; |
| return dt; |
| } |
| |
| CPDFSDK_DateTime& CPDFSDK_DateTime::AddDays(short days) { |
| if (days == 0) |
| return *this; |
| |
| int16_t y = dt.year, yy; |
| uint8_t m = dt.month; |
| uint8_t d = dt.day; |
| int mdays, ydays, ldays; |
| |
| ldays = days; |
| if (ldays > 0) { |
| yy = y; |
| if (((uint16_t)m * 100 + d) > 300) |
| yy++; |
| ydays = gAfxGetYearDays(yy); |
| 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; |
| yy = y; |
| if (((uint16_t)m * 100 + d) < 300) |
| yy--; |
| ydays = gAfxGetYearDays(yy); |
| while (ldays >= ydays) { |
| y--; |
| ldays -= ydays; |
| yy--; |
| mdays = gAfxGetMonthDays(y, m); |
| if (d > mdays) { |
| m++; |
| d -= mdays; |
| } |
| ydays = gAfxGetYearDays(yy); |
| } |
| while (ldays >= d) { |
| ldays -= d; |
| m--; |
| mdays = gAfxGetMonthDays(y, m); |
| d = mdays; |
| } |
| 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 = (uint8_t)(n / 3600); |
| dt.hour %= 24; |
| n %= 3600; |
| dt.minute = (uint8_t)(n / 60); |
| dt.second = (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_BAAnnot::CPDFSDK_BAAnnot(CPDF_Annot* pAnnot, |
| CPDFSDK_PageView* pPageView) |
| : CPDFSDK_Annot(pPageView), m_pAnnot(pAnnot) {} |
| |
| 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; |
| } |
| |
| // Tab Order |
| 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") != NULL; |
| } |
| |
| 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_FLOAT CPDFSDK_Annot::GetMinWidth() const { |
| return kMinWidth; |
| } |
| |
| FX_FLOAT CPDFSDK_Annot::GetMinHeight() const { |
| return kMinHeight; |
| } |
| |
| 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(); |
| } |
| |
| #ifdef PDF_ENABLE_XFA |
| FX_BOOL CPDFSDK_BAAnnot::IsXFAField() { |
| return FALSE; |
| } |
| #endif // PDF_ENABLE_XFA |
| |
| 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, NULL); |
| } |
| |
| 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 |