Revert "Remove IPDFSDK_AnnotHandler interface."
[pdfium.git] / fpdfsdk / src / fsdk_baseannot.cpp
index ac36e3c..c6731e6 100644 (file)
-// Copyright 2014 PDFium Authors. All rights reserved.\r
-// Use of this source code is governed by a BSD-style license that can be\r
-// found in the LICENSE file.\r
\r
-// Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com\r
-\r
-#include "../include/fsdk_define.h"\r
-#include "../include/fsdk_mgr.h"\r
-#include "../include/fsdk_baseannot.h"\r
-\r
-\r
-//---------------------------------------------------------------------------\r
-//                                                             CPDFSDK_DateTime        \r
-//---------------------------------------------------------------------------\r
-int _gAfxGetTimeZoneInSeconds(FX_CHAR tzhour, FX_BYTE tzminute)\r
-{\r
-       return (int)tzhour * 3600 + (int)tzminute * (tzhour >= 0 ? 60 : -60);\r
-}\r
-\r
-FX_BOOL _gAfxIsLeapYear(FX_SHORT year)\r
-{\r
-       return ((year % 400 == 0) || ((year % 4 == 0) && (year % 100 != 0)));\r
-}\r
-\r
-FX_WORD _gAfxGetYearDays(FX_SHORT year)\r
-{\r
-       return (_gAfxIsLeapYear(year) == TRUE ? 366 : 365);\r
-}\r
-\r
-FX_BYTE _gAfxGetMonthDays(FX_SHORT year, FX_BYTE month)\r
-{\r
-       FX_BYTE mDays;\r
-       switch (month)\r
-       {\r
-       case 1:\r
-       case 3:\r
-       case 5:\r
-       case 7:\r
-       case 8:\r
-       case 10:\r
-       case 12:\r
-               mDays = 31;\r
-               break;\r
-\r
-       case 4:\r
-       case 6:\r
-       case 9:\r
-       case 11:\r
-               mDays = 30;\r
-               break;\r
-\r
-       case 2:\r
-               if (_gAfxIsLeapYear(year) == TRUE)\r
-                       mDays = 29;\r
-               else\r
-                       mDays = 28;\r
-               break;\r
-\r
-       default:\r
-               mDays = 0;\r
-               break;\r
-       }\r
-\r
-       return mDays;\r
-}\r
-\r
-CPDFSDK_DateTime::CPDFSDK_DateTime()\r
-{\r
-       ResetDateTime();\r
-}\r
-\r
-CPDFSDK_DateTime::CPDFSDK_DateTime(const CFX_ByteString& dtStr)\r
-{\r
-       ResetDateTime();\r
-\r
-       FromPDFDateTimeString(dtStr);\r
-}\r
-\r
-CPDFSDK_DateTime::CPDFSDK_DateTime(const CPDFSDK_DateTime& datetime)\r
-{\r
-       operator = (datetime);\r
-}\r
-\r
-CPDFSDK_DateTime::CPDFSDK_DateTime(const FX_SYSTEMTIME& st)\r
-{\r
-       operator = (st) ;\r
-}\r
-\r
-\r
-void CPDFSDK_DateTime::ResetDateTime()\r
-{\r
-       tzset();\r
-\r
-       time_t  curTime;\r
-       time(&curTime);\r
-       struct tm* newtime;\r
-       //newtime = gmtime(&curTime);\r
-       newtime = localtime(&curTime);\r
-\r
-       dt.year = newtime->tm_year + 1900;\r
-       dt.month = newtime->tm_mon + 1;\r
-       dt.day = newtime->tm_mday;\r
-       dt.hour = newtime->tm_hour;\r
-       dt.minute = newtime->tm_min;\r
-       dt.second = newtime->tm_sec;\r
-//     dt.tzHour = _timezone / 3600 * -1;\r
-//     dt.tzMinute = (abs(_timezone) % 3600) / 60;\r
-}\r
-\r
-CPDFSDK_DateTime& CPDFSDK_DateTime::operator = (const CPDFSDK_DateTime& datetime)\r
-{\r
-       FXSYS_memcpy(&dt, &datetime.dt, sizeof(FX_DATETIME));\r
-       return *this;\r
-}\r
-\r
-CPDFSDK_DateTime& CPDFSDK_DateTime::operator = (const FX_SYSTEMTIME& st)\r
-{\r
-       tzset();\r
-\r
-       dt.year = (FX_SHORT)st.wYear;\r
-       dt.month = (FX_BYTE)st.wMonth;\r
-       dt.day = (FX_BYTE)st.wDay;\r
-       dt.hour = (FX_BYTE)st.wHour;\r
-       dt.minute = (FX_BYTE)st.wMinute;\r
-       dt.second = (FX_BYTE)st.wSecond;\r
-//     dt.tzHour = _timezone / 3600 * -1;\r
-//     dt.tzMinute = (abs(_timezone) % 3600) / 60;\r
-       return *this;\r
-}\r
-\r
-FX_BOOL CPDFSDK_DateTime::operator == (CPDFSDK_DateTime& datetime)\r
-{\r
-       return (FXSYS_memcmp(&dt, &datetime.dt, sizeof(FX_DATETIME)) == 0);\r
-}\r
-\r
-FX_BOOL CPDFSDK_DateTime::operator != (CPDFSDK_DateTime& datetime)\r
-{\r
-       return (FXSYS_memcmp(&dt, &datetime.dt, sizeof(FX_DATETIME)) != 0);\r
-}\r
-\r
-FX_BOOL CPDFSDK_DateTime::operator > (CPDFSDK_DateTime& datetime)\r
-{\r
-       CPDFSDK_DateTime dt1 = ToGMT();\r
-       CPDFSDK_DateTime dt2 = datetime.ToGMT();\r
-       int d1 = (((int)dt1.dt.year) << 16) | (((int)dt1.dt.month) << 8) | (int)dt1.dt.day;\r
-       int d2 = (((int)dt1.dt.hour) << 16) | (((int)dt1.dt.minute) << 8) | (int)dt1.dt.second;\r
-       int d3 = (((int)dt2.dt.year) << 16) | (((int)dt2.dt.month) << 8) | (int)dt2.dt.day;\r
-       int d4 = (((int)dt2.dt.hour) << 16) | (((int)dt2.dt.minute) << 8) | (int)dt2.dt.second;\r
-\r
-       if (d1 > d3) return TRUE;\r
-       if (d2 > d4) return TRUE;\r
-       return FALSE;\r
-}\r
-\r
-FX_BOOL CPDFSDK_DateTime::operator >= (CPDFSDK_DateTime& datetime)\r
-{\r
-       CPDFSDK_DateTime dt1 = ToGMT();\r
-       CPDFSDK_DateTime dt2 = datetime.ToGMT();\r
-       int d1 = (((int)dt1.dt.year) << 16) | (((int)dt1.dt.month) << 8) | (int)dt1.dt.day;\r
-       int d2 = (((int)dt1.dt.hour) << 16) | (((int)dt1.dt.minute) << 8) | (int)dt1.dt.second;\r
-       int d3 = (((int)dt2.dt.year) << 16) | (((int)dt2.dt.month) << 8) | (int)dt2.dt.day;\r
-       int d4 = (((int)dt2.dt.hour) << 16) | (((int)dt2.dt.minute) << 8) | (int)dt2.dt.second;\r
-\r
-       if (d1 >= d3) return TRUE;\r
-       if (d2 >= d4) return TRUE;\r
-       return FALSE;\r
-}\r
-\r
-FX_BOOL CPDFSDK_DateTime::operator < (CPDFSDK_DateTime& datetime)\r
-{\r
-       CPDFSDK_DateTime dt1 = ToGMT();\r
-       CPDFSDK_DateTime dt2 = datetime.ToGMT();\r
-       int d1 = (((int)dt1.dt.year) << 16) | (((int)dt1.dt.month) << 8) | (int)dt1.dt.day;\r
-       int d2 = (((int)dt1.dt.hour) << 16) | (((int)dt1.dt.minute) << 8) | (int)dt1.dt.second;\r
-       int d3 = (((int)dt2.dt.year) << 16) | (((int)dt2.dt.month) << 8) | (int)dt2.dt.day;\r
-       int d4 = (((int)dt2.dt.hour) << 16) | (((int)dt2.dt.minute) << 8) | (int)dt2.dt.second;\r
-\r
-       if (d1 < d3) return TRUE;\r
-       if (d2 < d4) return TRUE;\r
-       return FALSE;\r
-}\r
-\r
-FX_BOOL CPDFSDK_DateTime::operator <= (CPDFSDK_DateTime& datetime)\r
-{\r
-       CPDFSDK_DateTime dt1 = ToGMT();\r
-       CPDFSDK_DateTime dt2 = datetime.ToGMT();\r
-       int d1 = (((int)dt1.dt.year) << 16) | (((int)dt1.dt.month) << 8) | (int)dt1.dt.day;\r
-       int d2 = (((int)dt1.dt.hour) << 16) | (((int)dt1.dt.minute) << 8) | (int)dt1.dt.second;\r
-       int d3 = (((int)dt2.dt.year) << 16) | (((int)dt2.dt.month) << 8) | (int)dt2.dt.day;\r
-       int d4 = (((int)dt2.dt.hour) << 16) | (((int)dt2.dt.minute) << 8) | (int)dt2.dt.second;\r
-\r
-       if (d1 <= d3) return TRUE;\r
-       if (d2 <= d4) return TRUE;\r
-       return FALSE;\r
-}\r
-\r
-CPDFSDK_DateTime::operator time_t()\r
-{\r
-       struct tm newtime;\r
-\r
-       newtime.tm_year = dt.year - 1900;\r
-       newtime.tm_mon = dt.month - 1;\r
-       newtime.tm_mday = dt.day;\r
-       newtime.tm_hour = dt.hour;\r
-       newtime.tm_min = dt.minute;\r
-       newtime.tm_sec = dt.second;\r
-\r
-       return mktime(&newtime);\r
-}\r
-\r
-CPDFSDK_DateTime& CPDFSDK_DateTime::FromPDFDateTimeString(const CFX_ByteString& dtStr)\r
-{\r
-       int strLength = dtStr.GetLength();\r
-       if (strLength > 0)\r
-       {\r
-               int i = 0;\r
-               int j, k;\r
-               FX_CHAR ch;\r
-               while (i < strLength)\r
-               {\r
-                       ch = dtStr[i];\r
-                       if (ch >= '0' && ch <= '9') break;\r
-                       i ++;\r
-               }\r
-               if (i >= strLength) return *this;\r
-\r
-               j = 0;\r
-               k = 0;\r
-               while (i < strLength && j < 4)\r
-               {\r
-                       ch = dtStr[i];\r
-                       k = k * 10 + ch - '0';\r
-                       j ++;\r
-                       if (ch < '0' || ch > '9') break;\r
-                       i ++;\r
-               }\r
-               dt.year = (FX_SHORT)k;\r
-               if (i >= strLength || j < 4) return *this;\r
-\r
-               j = 0;\r
-               k = 0;\r
-               while (i < strLength && j < 2)\r
-               {\r
-                       ch = dtStr[i];\r
-                       k = k * 10 + ch - '0';\r
-                       j ++;\r
-                       if (ch < '0' || ch > '9') break;\r
-                       i ++;\r
-               }\r
-               dt.month = (FX_BYTE)k;\r
-               if (i >= strLength || j < 2) return *this;\r
-\r
-               j = 0;\r
-               k = 0;\r
-               while (i < strLength && j < 2)\r
-               {\r
-                       ch = dtStr[i];\r
-                       k = k * 10 + ch - '0';\r
-                       j ++;\r
-                       if (ch < '0' || ch > '9') break;\r
-                       i ++;\r
-               }\r
-               dt.day = (FX_BYTE)k;\r
-               if (i >= strLength || j < 2) return *this;\r
-\r
-               j = 0;\r
-               k = 0;\r
-               while (i < strLength && j < 2)\r
-               {\r
-                       ch = dtStr[i];\r
-                       k = k * 10 + ch - '0';\r
-                       j ++;\r
-                       if (ch < '0' || ch > '9') break;\r
-                       i ++;\r
-               }\r
-               dt.hour = (FX_BYTE)k;\r
-               if (i >= strLength || j < 2) return *this;\r
-\r
-               j = 0;\r
-               k = 0;\r
-               while (i < strLength && j < 2)\r
-               {\r
-                       ch = dtStr[i];\r
-                       k = k * 10 + ch - '0';\r
-                       j ++;\r
-                       if (ch < '0' || ch > '9') break;\r
-                       i ++;\r
-               }\r
-               dt.minute = (FX_BYTE)k;\r
-               if (i >= strLength || j < 2) return *this;\r
-\r
-               j = 0;\r
-               k = 0;\r
-               while (i < strLength && j < 2)\r
-               {\r
-                       ch = dtStr[i];\r
-                       k = k * 10 + ch - '0';\r
-                       j ++;\r
-                       if (ch < '0' || ch > '9') break;\r
-                       i ++;\r
-               }\r
-               dt.second = (FX_BYTE)k;\r
-               if (i >= strLength || j < 2) return *this;\r
-\r
-               ch = dtStr[i ++];\r
-               if (ch != '-' && ch != '+') return *this;\r
-               if (ch == '-')\r
-                       dt.tzHour = -1;\r
-               else\r
-                       dt.tzHour = 1;\r
-               j = 0;\r
-               k = 0;\r
-               while (i < strLength && j < 2)\r
-               {\r
-                       ch = dtStr[i];\r
-                       k = k * 10 + ch - '0';\r
-                       j ++;\r
-                       if (ch < '0' || ch > '9') break;\r
-                       i ++;\r
-               }\r
-               dt.tzHour *= (FX_CHAR)k;\r
-               if (i >= strLength || j < 2) return *this;\r
-\r
-               ch = dtStr[i ++];\r
-               if (ch != '\'') return *this;\r
-               j = 0;\r
-               k = 0;\r
-               while (i < strLength && j < 2)\r
-               {\r
-                       ch = dtStr[i];\r
-                       k = k * 10 + ch - '0';\r
-                       j ++;\r
-                       if (ch < '0' || ch > '9') break;\r
-                       i ++;\r
-               }\r
-               dt.tzMinute = (FX_BYTE)k;\r
-               if (i >= strLength || j < 2) return *this;\r
-       }\r
-\r
-       return  *this;\r
-}\r
-\r
-CFX_ByteString CPDFSDK_DateTime::ToCommonDateTimeString()\r
-{\r
-       CFX_ByteString str1;\r
-       str1.Format("%04d-%02d-%02d %02d:%02d:%02d ", dt.year, dt.month, dt.day, dt.hour, dt.minute, dt.second);\r
-       if (dt.tzHour < 0)\r
-               str1 += "-";\r
-       else\r
-               str1 += "+";\r
-       CFX_ByteString str2;\r
-       str2.Format("%02d:%02d", abs(dt.tzHour), dt.tzMinute);\r
-       return str1 + str2;\r
-}\r
-\r
-CFX_ByteString CPDFSDK_DateTime::ToPDFDateTimeString()\r
-{\r
-       CFX_ByteString dtStr;\r
-       char tempStr[32];\r
-       sprintf(tempStr, "D:%04d%02d%02d%02d%02d%02d", dt.year, dt.month, dt.day, dt.hour, dt.minute, dt.second);\r
-       dtStr = CFX_ByteString(tempStr);\r
-       if (dt.tzHour < 0)\r
-               dtStr += CFX_ByteString("-");\r
-       else\r
-               dtStr += CFX_ByteString("+");\r
-       sprintf(tempStr, "%02d'%02d'", abs(dt.tzHour), dt.tzMinute);\r
-       dtStr += CFX_ByteString(tempStr);\r
-       return dtStr;\r
-}\r
-\r
-void CPDFSDK_DateTime::ToSystemTime(FX_SYSTEMTIME& st)\r
-{\r
-       CPDFSDK_DateTime dt = *this;\r
-       time_t t = (time_t)dt;\r
-       struct tm* pTime = localtime(&t);\r
-       if(pTime){ \r
-               st.wYear = (FX_WORD)pTime->tm_year + 1900;\r
-               st.wMonth = (FX_WORD)pTime->tm_mon + 1;\r
-               st.wDay = (FX_WORD)pTime->tm_mday;\r
-               st.wDayOfWeek = (FX_WORD)pTime->tm_wday;\r
-               st.wHour = (FX_WORD)pTime->tm_hour;\r
-               st.wMinute = (FX_WORD)pTime->tm_min;\r
-               st.wSecond = (FX_WORD)pTime->tm_sec;\r
-               st.wMilliseconds = 0;\r
-       }\r
-}\r
-\r
-CPDFSDK_DateTime CPDFSDK_DateTime::ToGMT()\r
-{\r
-       CPDFSDK_DateTime dt = *this;\r
-       dt.AddSeconds(-_gAfxGetTimeZoneInSeconds(dt.dt.tzHour, dt.dt.tzMinute));\r
-       dt.dt.tzHour = 0;\r
-       dt.dt.tzMinute = 0;\r
-       return dt;\r
-}\r
-\r
-CPDFSDK_DateTime& CPDFSDK_DateTime::AddDays(short days)\r
-{\r
-       if (days == 0) return *this;\r
-\r
-       FX_SHORT        y = dt.year, yy;\r
-       FX_BYTE         m = dt.month;\r
-       FX_BYTE         d = dt.day;\r
-       int                     mdays, ydays, ldays;\r
-\r
-       ldays = days;\r
-       if (ldays > 0)\r
-       {\r
-               yy = y;\r
-               if (((FX_WORD)m * 100 + d) > 300) yy ++;\r
-               ydays = _gAfxGetYearDays(yy);\r
-               while (ldays >= ydays)\r
-               {\r
-                       y ++;\r
-                       ldays -= ydays;\r
-                       yy ++;\r
-                       mdays = _gAfxGetMonthDays(y, m);\r
-                       if (d > mdays)\r
-                       {\r
-                               m ++;\r
-                               d -= mdays;\r
-                       }\r
-                       ydays = _gAfxGetYearDays(yy);\r
-               }\r
-               mdays = _gAfxGetMonthDays(y, m) - d + 1;\r
-               while (ldays >= mdays)\r
-               {\r
-                       ldays -= mdays;\r
-                       m ++;\r
-                       d = 1;\r
-                       mdays = _gAfxGetMonthDays(y, m);\r
-               }\r
-               d += ldays;\r
-       }\r
-       else\r
-       {\r
-               ldays *= -1;\r
-               yy = y;\r
-               if (((FX_WORD)m * 100 + d) < 300) yy --;\r
-               ydays = _gAfxGetYearDays(yy);\r
-               while (ldays >= ydays)\r
-               {\r
-                       y --;\r
-                       ldays -= ydays;\r
-                       yy --;\r
-                       mdays = _gAfxGetMonthDays(y, m);\r
-                       if (d > mdays)\r
-                       {\r
-                               m ++;\r
-                               d -= mdays;\r
-                       }\r
-                       ydays = _gAfxGetYearDays(yy);\r
-               }\r
-               while (ldays >= d)\r
-               {\r
-                       ldays -= d;\r
-                       m --;\r
-                       mdays = _gAfxGetMonthDays(y, m);\r
-                       d = mdays;\r
-               }\r
-               d -= ldays;\r
-       }\r
-\r
-       dt.year = y;\r
-       dt.month = m;\r
-       dt.day = d;\r
-\r
-       return *this;\r
-}\r
-\r
-CPDFSDK_DateTime& CPDFSDK_DateTime::AddSeconds(int seconds)\r
-{\r
-       if (seconds == 0) return *this;\r
-\r
-       int     n;\r
-       int     days;\r
-\r
-       n = dt.hour * 3600 + dt.minute * 60 + dt.second + seconds;\r
-       if (n < 0)\r
-       {\r
-               days = (n - 86399) / 86400;\r
-               n -= days * 86400;\r
-       }\r
-       else\r
-       {\r
-               days = n / 86400;\r
-               n %= 86400;\r
-       }\r
-       dt.hour = (FX_BYTE)(n / 3600);\r
-       dt.hour %= 24;\r
-       n %= 3600;\r
-       dt.minute = (FX_BYTE)(n / 60);\r
-       dt.second = (FX_BYTE)(n % 60);\r
-       if (days != 0) AddDays(days);\r
-\r
-       return *this;\r
-}\r
-\r
-\r
-//---------------------------------------------------------------------------\r
-//                                                             CPDFSDK_Annot   \r
-//---------------------------------------------------------------------------\r
-CPDFSDK_Annot::CPDFSDK_Annot(CPDF_Annot* pAnnot, CPDFSDK_PageView* pPageView) :\r
-m_pAnnot(pAnnot),\r
-m_pPageView(pPageView),\r
-m_bSelected(FALSE),\r
-m_nTabOrder(-1)\r
-{\r
-}\r
-\r
-CPDFSDK_Annot::~CPDFSDK_Annot()\r
-{\r
-       m_pAnnot = NULL;\r
-       m_pPageView = NULL;\r
-}\r
-\r
-CPDF_Annot*    CPDFSDK_Annot::GetPDFAnnot()\r
-{\r
-       return m_pAnnot;\r
-}\r
-\r
-FX_DWORD CPDFSDK_Annot::GetFlags()\r
-{\r
-       ASSERT(m_pAnnot != NULL);\r
-       \r
-       return m_pAnnot->GetFlags();\r
-}\r
-\r
-void CPDFSDK_Annot::SetPage(CPDFSDK_PageView* pPageView)\r
-{\r
-       m_pPageView = pPageView;\r
-}\r
-\r
-CPDFSDK_PageView* CPDFSDK_Annot::GetPageView()\r
-{\r
-       return m_pPageView;\r
-}\r
-\r
-FX_BOOL CPDFSDK_Annot::IsSelected()\r
-{\r
-       return m_bSelected;\r
-}\r
-\r
-void CPDFSDK_Annot::SetSelected(FX_BOOL bSelected)\r
-{\r
-       m_bSelected = bSelected;\r
-}\r
-\r
-// Tab Order   \r
-int CPDFSDK_Annot::GetTabOrder()\r
-{\r
-       return m_nTabOrder;\r
-}\r
-\r
-void CPDFSDK_Annot::SetTabOrder(int iTabOrder)\r
-{\r
-       m_nTabOrder = iTabOrder;\r
-}\r
-\r
-CPDF_Dictionary* CPDFSDK_Annot::GetAnnotDict() const\r
-{\r
-       ASSERT(m_pAnnot != NULL);\r
-       \r
-       return m_pAnnot->m_pAnnotDict;\r
-}\r
-\r
-void CPDFSDK_Annot::SetRect(const CPDF_Rect& rect)\r
-{\r
-       ASSERT(m_pAnnot != NULL);\r
-       ASSERT(m_pAnnot->m_pAnnotDict != NULL);\r
-       ASSERT(rect.right - rect.left >= GetMinWidth());\r
-       ASSERT(rect.top - rect.bottom >= GetMinHeight());\r
-       \r
-       m_pAnnot->m_pAnnotDict->SetAtRect("Rect", rect);\r
-}\r
-\r
-CPDF_Rect CPDFSDK_Annot::GetRect() const\r
-{\r
-       ASSERT(m_pAnnot != NULL);\r
-       \r
-       CPDF_Rect rect;\r
-       m_pAnnot->GetRect(rect);\r
-       \r
-       return rect;\r
-}\r
-\r
-CFX_ByteString CPDFSDK_Annot::GetType() const\r
-{\r
-       ASSERT(m_pAnnot != NULL);\r
-       \r
-       return m_pAnnot->GetSubType();\r
-}\r
-\r
-CFX_ByteString CPDFSDK_Annot::GetSubType() const\r
-{\r
-       return "";\r
-}\r
-\r
-void CPDFSDK_Annot::ResetAppearance()\r
-{\r
-       ASSERT(FALSE);\r
-}\r
-\r
-void CPDFSDK_Annot::DrawAppearance(CFX_RenderDevice* pDevice, const CPDF_Matrix* pUser2Device,\r
-                                                                  CPDF_Annot::AppearanceMode mode, const CPDF_RenderOptions* pOptions) \r
-{\r
-       ASSERT(m_pPageView != NULL);\r
-       ASSERT(m_pAnnot != NULL);\r
-       \r
-       m_pAnnot->DrawAppearance(m_pPageView->GetPDFPage(), pDevice, pUser2Device, mode, pOptions);\r
-}\r
-\r
-FX_BOOL        CPDFSDK_Annot::IsAppearanceValid()\r
-{\r
-       ASSERT(m_pAnnot != NULL);\r
-       ASSERT(m_pAnnot->m_pAnnotDict != NULL);\r
-       \r
-       return m_pAnnot->m_pAnnotDict->GetDict("AP") != NULL;\r
-}\r
-\r
-FX_BOOL        CPDFSDK_Annot::IsAppearanceValid(CPDF_Annot::AppearanceMode mode)\r
-{\r
-       ASSERT(m_pAnnot != NULL);\r
-       ASSERT(m_pAnnot->m_pAnnotDict != NULL);\r
-       \r
-       CPDF_Dictionary* pAP = m_pAnnot->m_pAnnotDict->GetDict("AP");\r
-       if (pAP == NULL) return FALSE;\r
-       \r
-       // Choose the right sub-ap\r
-       const FX_CHAR* ap_entry = "N";\r
-       if (mode == CPDF_Annot::Down)\r
-               ap_entry = "D";\r
-       else if (mode == CPDF_Annot::Rollover)\r
-               ap_entry = "R";\r
-       if (!pAP->KeyExist(ap_entry))\r
-               ap_entry = "N";\r
-       \r
-       // Get the AP stream or subdirectory\r
-       CPDF_Object* psub = pAP->GetElementValue(ap_entry);\r
-       if (psub == NULL) return FALSE;\r
-       \r
-       return TRUE;\r
-}\r
-\r
-void CPDFSDK_Annot::DrawBorder(CFX_RenderDevice* pDevice, const CPDF_Matrix* pUser2Device,\r
-                                                  const CPDF_RenderOptions* pOptions)\r
-{\r
-       ASSERT(m_pAnnot != NULL);\r
-       m_pAnnot->DrawBorder(pDevice, pUser2Device, pOptions); \r
-}\r
-\r
-void CPDFSDK_Annot::ClearCachedAP()\r
-{\r
-       ASSERT(m_pAnnot != NULL);\r
-       m_pAnnot->ClearCachedAP();\r
-}    \r
-\r
-void CPDFSDK_Annot::SetContents(const CFX_WideString& sContents)\r
-{\r
-       ASSERT(m_pAnnot != NULL);\r
-       ASSERT(m_pAnnot->m_pAnnotDict != NULL);\r
-       \r
-       if (sContents.IsEmpty())\r
-               m_pAnnot->m_pAnnotDict->RemoveAt("Contents");\r
-       else\r
-               m_pAnnot->m_pAnnotDict->SetAtString("Contents", PDF_EncodeText(sContents));\r
-}\r
-\r
-CFX_WideString CPDFSDK_Annot::GetContents() const\r
-{\r
-       ASSERT(m_pAnnot != NULL);\r
-       ASSERT(m_pAnnot->m_pAnnotDict != NULL);\r
-       \r
-       return m_pAnnot->m_pAnnotDict->GetUnicodeText("Contents");\r
-}\r
-\r
-void CPDFSDK_Annot::SetAnnotName(const CFX_WideString& sName)\r
-{\r
-       ASSERT(m_pAnnot != NULL);\r
-       ASSERT(m_pAnnot->m_pAnnotDict != NULL);\r
-       \r
-       if (sName.IsEmpty())\r
-               m_pAnnot->m_pAnnotDict->RemoveAt("NM");\r
-       else\r
-               m_pAnnot->m_pAnnotDict->SetAtString("NM", PDF_EncodeText(sName));\r
-}\r
-\r
-CFX_WideString CPDFSDK_Annot::GetAnnotName() const\r
-{\r
-       ASSERT(m_pAnnot != NULL);\r
-       ASSERT(m_pAnnot->m_pAnnotDict != NULL);\r
-       \r
-       return m_pAnnot->m_pAnnotDict->GetUnicodeText("NM");\r
-}\r
-\r
-void CPDFSDK_Annot::SetModifiedDate(const FX_SYSTEMTIME& st)\r
-{\r
-       ASSERT(m_pAnnot != NULL);\r
-       ASSERT(m_pAnnot->m_pAnnotDict != NULL);\r
-       \r
-       CPDFSDK_DateTime dt(st);\r
-       CFX_ByteString str = dt.ToPDFDateTimeString();\r
-       \r
-       if (str.IsEmpty())\r
-               m_pAnnot->m_pAnnotDict->RemoveAt("M");\r
-       else\r
-               m_pAnnot->m_pAnnotDict->SetAtString("M", str);\r
-}\r
-\r
-FX_SYSTEMTIME CPDFSDK_Annot::GetModifiedDate() const\r
-{\r
-       ASSERT(m_pAnnot != NULL);\r
-       ASSERT(m_pAnnot->m_pAnnotDict != NULL);\r
-       \r
-       FX_SYSTEMTIME systime;  \r
-       CFX_ByteString str = m_pAnnot->m_pAnnotDict->GetString("M");\r
-       \r
-       CPDFSDK_DateTime dt(str);\r
-       dt.ToSystemTime(systime);\r
-       \r
-       return systime;\r
-}\r
-\r
-void CPDFSDK_Annot::SetFlags(int nFlags)\r
-{\r
-       ASSERT(m_pAnnot != NULL);\r
-       ASSERT(m_pAnnot->m_pAnnotDict != NULL);\r
-       \r
-       m_pAnnot->m_pAnnotDict->SetAtInteger("F", nFlags);\r
-}\r
-\r
-int CPDFSDK_Annot::GetFlags() const\r
-{\r
-       ASSERT(m_pAnnot != NULL);\r
-       ASSERT(m_pAnnot->m_pAnnotDict != NULL);\r
-       \r
-       return m_pAnnot->m_pAnnotDict->GetInteger("F");\r
-}\r
-\r
-void CPDFSDK_Annot::SetAppState(const CFX_ByteString& str)\r
-{\r
-       ASSERT(m_pAnnot != NULL);\r
-       ASSERT(m_pAnnot->m_pAnnotDict != NULL);\r
-\r
-       if (str.IsEmpty())\r
-               m_pAnnot->m_pAnnotDict->RemoveAt("AS");\r
-       else\r
-               m_pAnnot->m_pAnnotDict->SetAtString("AS", str);\r
-}\r
-\r
-CFX_ByteString CPDFSDK_Annot::GetAppState() const\r
-{\r
-       ASSERT(m_pAnnot != NULL);\r
-       ASSERT(m_pAnnot->m_pAnnotDict != NULL);\r
-\r
-       return m_pAnnot->m_pAnnotDict->GetString("AS");\r
-}\r
-\r
-void CPDFSDK_Annot::SetStructParent(int key)\r
-{\r
-       ASSERT(m_pAnnot != NULL);\r
-       ASSERT(m_pAnnot->m_pAnnotDict != NULL);\r
-       \r
-       m_pAnnot->m_pAnnotDict->SetAtInteger("StructParent", key);\r
-}\r
-\r
-int    CPDFSDK_Annot::GetStructParent() const\r
-{\r
-       ASSERT(m_pAnnot != NULL);\r
-       ASSERT(m_pAnnot->m_pAnnotDict != NULL);\r
-       \r
-       return m_pAnnot->m_pAnnotDict->GetInteger("StructParent");\r
-}\r
-\r
-//border\r
-void CPDFSDK_Annot::SetBorderWidth(int nWidth)\r
-{\r
-       ASSERT(m_pAnnot != NULL);\r
-       ASSERT(m_pAnnot->m_pAnnotDict != NULL);\r
-\r
-       CPDF_Array* pBorder = m_pAnnot->m_pAnnotDict->GetArray("Border");\r
-\r
-       if (pBorder)\r
-       {\r
-               pBorder->SetAt(2, FX_NEW CPDF_Number(nWidth));\r
-       }\r
-       else\r
-       {\r
-               CPDF_Dictionary* pBSDict = m_pAnnot->m_pAnnotDict->GetDict("BS");\r
-\r
-               if (!pBSDict)\r
-               {\r
-                       pBSDict = FX_NEW CPDF_Dictionary;\r
-                       m_pAnnot->m_pAnnotDict->SetAt("BS", pBSDict);\r
-               }\r
-\r
-               pBSDict->SetAtInteger("W", nWidth);\r
-       }\r
-}\r
-\r
-int    CPDFSDK_Annot::GetBorderWidth() const\r
-{\r
-       ASSERT(m_pAnnot != NULL);\r
-       ASSERT(m_pAnnot->m_pAnnotDict != NULL);\r
-\r
-       CPDF_Array* pBorder = m_pAnnot->m_pAnnotDict->GetArray("Border");\r
-\r
-       if (pBorder)\r
-       {\r
-               return pBorder->GetInteger(2);\r
-       }\r
-       else\r
-       {\r
-               CPDF_Dictionary* pBSDict = m_pAnnot->m_pAnnotDict->GetDict("BS");\r
-\r
-               if (pBSDict)\r
-               {\r
-                       return pBSDict->GetInteger("W", 1);\r
-               }\r
-       }\r
-       return 1;\r
-}\r
-\r
-void CPDFSDK_Annot::SetBorderStyle(int nStyle)\r
-{\r
-       ASSERT(m_pAnnot != NULL);\r
-       ASSERT(m_pAnnot->m_pAnnotDict != NULL);\r
-\r
-       CPDF_Dictionary* pBSDict = m_pAnnot->m_pAnnotDict->GetDict("BS");\r
-       if (!pBSDict)\r
-       {\r
-               pBSDict = FX_NEW CPDF_Dictionary;\r
-               m_pAnnot->m_pAnnotDict->SetAt("BS", pBSDict);\r
-       }\r
-\r
-       switch (nStyle)\r
-       {\r
-       case BBS_SOLID:\r
-               pBSDict->SetAtName("S", "S");\r
-               break;\r
-       case BBS_DASH:\r
-               pBSDict->SetAtName("S", "D");\r
-               break;\r
-       case BBS_BEVELED:\r
-               pBSDict->SetAtName("S", "B");\r
-               break;\r
-       case BBS_INSET:\r
-               pBSDict->SetAtName("S", "I");\r
-               break;\r
-       case BBS_UNDERLINE:\r
-               pBSDict->SetAtName("S", "U");\r
-               break;\r
-       }\r
-}\r
-\r
-int    CPDFSDK_Annot::GetBorderStyle() const\r
-{\r
-       ASSERT(m_pAnnot != NULL);\r
-       ASSERT(m_pAnnot->m_pAnnotDict != NULL);\r
-\r
-       CPDF_Dictionary* pBSDict = m_pAnnot->m_pAnnotDict->GetDict("BS");\r
-       if (pBSDict)\r
-       {\r
-               CFX_ByteString sBorderStyle = pBSDict->GetString("S", "S");\r
-               if (sBorderStyle == "S") return BBS_SOLID;\r
-               if (sBorderStyle == "D") return BBS_DASH;\r
-               if (sBorderStyle == "B") return BBS_BEVELED;\r
-               if (sBorderStyle == "I") return BBS_INSET;\r
-               if (sBorderStyle == "U") return BBS_UNDERLINE;\r
-       }\r
-\r
-       CPDF_Array* pBorder = m_pAnnot->m_pAnnotDict->GetArray("Border");\r
-       if (pBorder)\r
-       {\r
-               if (pBorder->GetCount() >= 4) \r
-               { \r
-                       CPDF_Array *pDP = pBorder->GetArray(3);\r
-                       if (pDP && pDP->GetCount() > 0)\r
-                               return BBS_DASH;\r
-               }\r
-       }\r
-\r
-       return BBS_SOLID;\r
-}\r
-\r
-void CPDFSDK_Annot::SetBorderDash(const CFX_IntArray& array)\r
-{\r
-       ASSERT(m_pAnnot != NULL);\r
-       ASSERT(m_pAnnot->m_pAnnotDict != NULL);\r
-\r
-       CPDF_Dictionary* pBSDict = m_pAnnot->m_pAnnotDict->GetDict("BS");\r
-       if (!pBSDict)\r
-       {\r
-               pBSDict = FX_NEW CPDF_Dictionary;\r
-               m_pAnnot->m_pAnnotDict->SetAt("BS", pBSDict);\r
-       }\r
-\r
-       CPDF_Array* pArray = FX_NEW CPDF_Array;\r
-       for (int i=0,sz=array.GetSize(); i<sz; i++)\r
-       {\r
-               pArray->AddInteger(array[i]);\r
-       }\r
-\r
-       pBSDict->SetAt("D", pArray);\r
-}\r
-\r
-void CPDFSDK_Annot::GetBorderDash(CFX_IntArray& array) const\r
-{\r
-       ASSERT(m_pAnnot != NULL);\r
-       ASSERT(m_pAnnot->m_pAnnotDict != NULL);\r
-\r
-       CPDF_Array* pDash = NULL;\r
-\r
-       CPDF_Array* pBorder = m_pAnnot->m_pAnnotDict->GetArray("Border");\r
-       if (pBorder)\r
-       {\r
-               pDash = pBorder->GetArray(3);\r
-       }\r
-       else\r
-       {\r
-               CPDF_Dictionary* pBSDict = m_pAnnot->m_pAnnotDict->GetDict("BS");\r
-               if (pBSDict)\r
-               {\r
-                       pDash = pBSDict->GetArray("D");\r
-               }\r
-       }\r
-\r
-       if (pDash)\r
-       {\r
-               for (int i=0,sz=pDash->GetCount(); i<sz; i++)\r
-               {\r
-                       array.Add(pDash->GetInteger(i));\r
-               }\r
-       }\r
-}\r
-\r
-void CPDFSDK_Annot::SetColor(FX_COLORREF color)\r
-{\r
-       ASSERT(m_pAnnot != NULL);\r
-       ASSERT(m_pAnnot->m_pAnnotDict != NULL);\r
-\r
-       CPDF_Array* pArray = FX_NEW CPDF_Array;\r
-       pArray->AddNumber((FX_FLOAT)FXSYS_GetRValue(color) / 255.0f);\r
-       pArray->AddNumber((FX_FLOAT)FXSYS_GetGValue(color) / 255.0f);\r
-       pArray->AddNumber((FX_FLOAT)FXSYS_GetBValue(color) / 255.0f);\r
-       m_pAnnot->m_pAnnotDict->SetAt("C", pArray);\r
-}\r
-\r
-void CPDFSDK_Annot::RemoveColor()\r
-{\r
-       ASSERT(m_pAnnot != NULL);\r
-       ASSERT(m_pAnnot->m_pAnnotDict != NULL);\r
-\r
-       m_pAnnot->m_pAnnotDict->RemoveAt("C") ; \r
-}\r
-\r
-FX_BOOL CPDFSDK_Annot::GetColor(FX_COLORREF& color) const\r
-{\r
-       ASSERT(m_pAnnot != NULL);\r
-       ASSERT(m_pAnnot->m_pAnnotDict != NULL);\r
-\r
-       if (CPDF_Array* pEntry = m_pAnnot->m_pAnnotDict->GetArray("C"))         \r
-       {\r
-               int nCount = pEntry->GetCount();\r
-               if (nCount == 1)\r
-               {\r
-                       FX_FLOAT g = pEntry->GetNumber(0) * 255;\r
-\r
-                       color = FXSYS_RGB((int)g, (int)g, (int)g);\r
-\r
-                       return TRUE;\r
-               }\r
-               else if (nCount == 3)\r
-               {\r
-                       FX_FLOAT r = pEntry->GetNumber(0) * 255;\r
-                       FX_FLOAT g = pEntry->GetNumber(1) * 255;\r
-                       FX_FLOAT b = pEntry->GetNumber(2) * 255;\r
-\r
-                       color = FXSYS_RGB((int)r, (int)g, (int)b);\r
-\r
-                       return TRUE;\r
-               }\r
-               else if (nCount == 4)\r
-               {\r
-                       FX_FLOAT c = pEntry->GetNumber(0);\r
-                       FX_FLOAT m = pEntry->GetNumber(1);\r
-                       FX_FLOAT y = pEntry->GetNumber(2);\r
-                       FX_FLOAT k = pEntry->GetNumber(3);\r
-\r
-                       FX_FLOAT r = 1.0f - FX_MIN(1.0f, c + k);\r
-                       FX_FLOAT g = 1.0f - FX_MIN(1.0f, m + k);\r
-                       FX_FLOAT b = 1.0f - FX_MIN(1.0f, y + k);\r
-\r
-                       color = FXSYS_RGB((int)(r * 255), (int)(g * 255), (int)(b * 255));\r
-\r
-                       return TRUE;\r
-               }\r
-       }\r
-\r
-       return FALSE;\r
-}\r
-\r
-\r
-void CPDFSDK_Annot::WriteAppearance(const CFX_ByteString& sAPType, const CPDF_Rect& rcBBox, \r
-                                                               const CPDF_Matrix& matrix, const CFX_ByteString& sContents,\r
-                                                               const CFX_ByteString& sAPState)\r
-{\r
-       ASSERT(m_pAnnot != NULL);\r
-       ASSERT(m_pAnnot->m_pAnnotDict != NULL);\r
-       \r
-       CPDF_Dictionary* pAPDict = m_pAnnot->m_pAnnotDict->GetDict("AP");\r
-       \r
-       if (!pAPDict) \r
-       {\r
-               pAPDict = FX_NEW CPDF_Dictionary;\r
-               m_pAnnot->m_pAnnotDict->SetAt("AP", pAPDict);\r
-       }\r
-       \r
-       CPDF_Stream* pStream = NULL;\r
-       CPDF_Dictionary* pParentDict = NULL;\r
-       \r
-       if (sAPState.IsEmpty())\r
-       {\r
-               pParentDict = pAPDict;\r
-               pStream = pAPDict->GetStream(sAPType);\r
-       }\r
-       else\r
-       {\r
-               CPDF_Dictionary* pAPTypeDict = pAPDict->GetDict(sAPType);\r
-               if (!pAPTypeDict)\r
-               {\r
-                       pAPTypeDict = FX_NEW CPDF_Dictionary;\r
-                       pAPDict->SetAt(sAPType, pAPTypeDict);\r
-               }\r
-               \r
-               pParentDict = pAPTypeDict;\r
-               pStream = pAPTypeDict->GetStream(sAPState);\r
-       }\r
-       \r
-       if (!pStream) \r
-       {\r
-               ASSERT(m_pPageView != NULL);\r
-               CPDF_Document* pDoc = m_pPageView->GetPDFDocument();\r
-               ASSERT(pDoc != NULL);\r
-               \r
-               pStream = FX_NEW CPDF_Stream(NULL, 0, NULL);\r
-               FX_INT32 objnum = pDoc->AddIndirectObject(pStream);\r
-               //pAPDict->SetAtReference(sAPType, pDoc, objnum);\r
-               ASSERT(pParentDict != NULL);\r
-               pParentDict->SetAtReference(sAPType, pDoc, objnum);\r
-       }\r
-       \r
-       CPDF_Dictionary * pStreamDict = pStream->GetDict();\r
-       \r
-       if (!pStreamDict)\r
-       {\r
-               pStreamDict = FX_NEW CPDF_Dictionary;\r
-               pStreamDict->SetAtName("Type", "XObject");\r
-               pStreamDict->SetAtName("Subtype", "Form");\r
-               pStreamDict->SetAtInteger("FormType", 1);\r
-               pStream->InitStream(NULL,0,pStreamDict);\r
-       }\r
-       \r
-       if (pStreamDict)\r
-       {\r
-               pStreamDict->SetAtMatrix("Matrix",matrix);      \r
-               pStreamDict->SetAtRect("BBox", rcBBox);         \r
-       }\r
-       \r
-       pStream->SetData((FX_BYTE*)(FX_LPCSTR)sContents, sContents.GetLength(), FALSE, FALSE);\r
-}\r
-\r
-#define BA_ANNOT_MINWIDTH                      1\r
-#define BA_ANNOT_MINHEIGHT                     1\r
-\r
-FX_FLOAT CPDFSDK_Annot::GetMinWidth() const\r
-{\r
-       return BA_ANNOT_MINWIDTH;\r
-}\r
-\r
-FX_FLOAT CPDFSDK_Annot::GetMinHeight() const\r
-{\r
-       return BA_ANNOT_MINHEIGHT;\r
-}\r
-\r
-FX_BOOL CPDFSDK_Annot::CreateFormFiller()\r
-{\r
-       return TRUE;\r
-}\r
-FX_BOOL        CPDFSDK_Annot::IsVisible() const\r
-{\r
-       int nFlags = GetFlags();\r
-       return !((nFlags & ANNOTFLAG_INVISIBLE) || (nFlags & ANNOTFLAG_HIDDEN) || (nFlags & ANNOTFLAG_NOVIEW));\r
-}\r
-\r
-CPDF_Action CPDFSDK_Annot::GetAction() const\r
-{\r
-       ASSERT(m_pAnnot != NULL);\r
-       ASSERT(m_pAnnot->m_pAnnotDict != NULL);\r
-       \r
-       return m_pAnnot->m_pAnnotDict->GetDict("A");\r
-}\r
-\r
-void CPDFSDK_Annot::SetAction(const CPDF_Action& action)\r
-{\r
-       ASSERT(m_pAnnot != NULL);\r
-       ASSERT(m_pAnnot->m_pAnnotDict != NULL);\r
-       \r
-       ASSERT(action != NULL);\r
-       \r
-       if ((CPDF_Action&)action != m_pAnnot->m_pAnnotDict->GetDict("A"))\r
-       {\r
-               CPDF_Document* pDoc = m_pPageView->GetPDFDocument();\r
-               ASSERT(pDoc != NULL);\r
-               \r
-               if (action.m_pDict && (action.m_pDict->GetObjNum() == 0))\r
-                       pDoc->AddIndirectObject(action.m_pDict); \r
-               m_pAnnot->m_pAnnotDict->SetAtReference("A", pDoc, action.m_pDict->GetObjNum());\r
-       }\r
-}\r
-\r
-void CPDFSDK_Annot::RemoveAction()\r
-{\r
-       ASSERT(m_pAnnot != NULL);\r
-       ASSERT(m_pAnnot->m_pAnnotDict != NULL);\r
-       \r
-       m_pAnnot->m_pAnnotDict->RemoveAt("A");\r
-}\r
-\r
-CPDF_AAction CPDFSDK_Annot::GetAAction() const\r
-{\r
-       ASSERT(m_pAnnot != NULL);\r
-       ASSERT(m_pAnnot->m_pAnnotDict != NULL);\r
-       \r
-       return m_pAnnot->m_pAnnotDict->GetDict("AA");\r
-}\r
-\r
-void CPDFSDK_Annot::SetAAction(const CPDF_AAction& aa)\r
-{\r
-       ASSERT(m_pAnnot != NULL);\r
-       ASSERT(m_pAnnot->m_pAnnotDict != NULL);\r
-       ASSERT(aa != NULL);\r
-       \r
-       if ((CPDF_AAction&)aa != m_pAnnot->m_pAnnotDict->GetDict("AA"))\r
-               m_pAnnot->m_pAnnotDict->SetAt("AA", (CPDF_AAction&)aa);\r
-}\r
-\r
-void CPDFSDK_Annot::RemoveAAction()\r
-{\r
-       ASSERT(m_pAnnot != NULL);\r
-       ASSERT(m_pAnnot->m_pAnnotDict != NULL);\r
-       \r
-       m_pAnnot->m_pAnnotDict->RemoveAt("AA");\r
-}\r
-\r
-CPDF_Action    CPDFSDK_Annot::GetAAction(CPDF_AAction::AActionType eAAT)\r
-{\r
-       CPDF_AAction AAction = GetAAction();\r
-       \r
-       if (AAction.ActionExist(eAAT))\r
-       {\r
-               return AAction.GetAction(eAAT);\r
-       }\r
-       else if (eAAT == CPDF_AAction::ButtonUp)\r
-       {\r
-               return GetAction();\r
-       }\r
-       \r
-       return NULL;\r
-}\r
-\r
-void  CPDFSDK_Annot::Annot_OnDraw(CFX_RenderDevice* pDevice, CPDF_Matrix* pUser2Device, CPDF_RenderOptions* pOptions)\r
-{\r
-       \r
-       m_pAnnot->GetAPForm(m_pPageView->GetPDFPage(), CPDF_Annot::Normal);\r
-       m_pAnnot->DrawAppearance(m_pPageView->GetPDFPage(), pDevice, pUser2Device, CPDF_Annot::Normal, NULL);\r
-\r
-       return ;\r
-}\r
-\r
-CPDF_Page* CPDFSDK_Annot::GetPDFPage()\r
-{\r
-       if(m_pPageView)\r
-               return m_pPageView->GetPDFPage();\r
-       return NULL;\r
-}\r
-\r
+// 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 "../include/fsdk_define.h"
+#include "../include/fsdk_mgr.h"
+#include "../include/fsdk_baseannot.h"
+
+//---------------------------------------------------------------------------
+//                              CPDFSDK_DateTime
+//---------------------------------------------------------------------------
+int _gAfxGetTimeZoneInSeconds(FX_CHAR tzhour, uint8_t tzminute) {
+  return (int)tzhour * 3600 + (int)tzminute * (tzhour >= 0 ? 60 : -60);
+}
+
+FX_BOOL _gAfxIsLeapYear(int16_t year) {
+  return ((year % 400 == 0) || ((year % 4 == 0) && (year % 100 != 0)));
+}
+
+FX_WORD _gAfxGetYearDays(int16_t year) {
+  return (_gAfxIsLeapYear(year) == TRUE ? 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) == TRUE)
+        mDays = 29;
+      else
+        mDays = 28;
+      break;
+
+    default:
+      mDays = 0;
+      break;
+  }
+
+  return mDays;
+}
+
+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;
+  // newtime = gmtime(&curTime);
+  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;
+  //  dt.tzHour = _timezone / 3600 * -1;
+  //  dt.tzMinute = (abs(_timezone) % 3600) / 60;
+}
+
+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;
+  //  dt.tzHour = _timezone / 3600 * -1;
+  //  dt.tzMinute = (abs(_timezone) % 3600) / 60;
+  return *this;
+}
+
+FX_BOOL CPDFSDK_DateTime::operator==(CPDFSDK_DateTime& datetime) {
+  return (FXSYS_memcmp(&dt, &datetime.dt, sizeof(FX_DATETIME)) == 0);
+}
+
+FX_BOOL CPDFSDK_DateTime::operator!=(CPDFSDK_DateTime& datetime) {
+  return (FXSYS_memcmp(&dt, &datetime.dt, sizeof(FX_DATETIME)) != 0);
+}
+
+FX_BOOL CPDFSDK_DateTime::operator>(CPDFSDK_DateTime& datetime) {
+  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;
+
+  if (d1 > d3)
+    return TRUE;
+  if (d2 > d4)
+    return TRUE;
+  return FALSE;
+}
+
+FX_BOOL CPDFSDK_DateTime::operator>=(CPDFSDK_DateTime& datetime) {
+  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;
+
+  if (d1 >= d3)
+    return TRUE;
+  if (d2 >= d4)
+    return TRUE;
+  return FALSE;
+}
+
+FX_BOOL CPDFSDK_DateTime::operator<(CPDFSDK_DateTime& datetime) {
+  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;
+
+  if (d1 < d3)
+    return TRUE;
+  if (d2 < d4)
+    return TRUE;
+  return FALSE;
+}
+
+FX_BOOL CPDFSDK_DateTime::operator<=(CPDFSDK_DateTime& datetime) {
+  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;
+
+  if (d1 <= d3)
+    return TRUE;
+  if (d2 <= d4)
+    return TRUE;
+  return FALSE;
+}
+
+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) {
+      ch = dtStr[i];
+      if (ch >= '0' && ch <= '9')
+        break;
+      i++;
+    }
+    if (i >= strLength)
+      return *this;
+
+    j = 0;
+    k = 0;
+    while (i < strLength && j < 4) {
+      ch = dtStr[i];
+      k = k * 10 + ch - '0';
+      j++;
+      if (ch < '0' || ch > '9')
+        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 + ch - '0';
+      j++;
+      if (ch < '0' || ch > '9')
+        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 + ch - '0';
+      j++;
+      if (ch < '0' || ch > '9')
+        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 + ch - '0';
+      j++;
+      if (ch < '0' || ch > '9')
+        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 + ch - '0';
+      j++;
+      if (ch < '0' || ch > '9')
+        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 + ch - '0';
+      j++;
+      if (ch < '0' || ch > '9')
+        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 + ch - '0';
+      j++;
+      if (ch < '0' || ch > '9')
+        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 + ch - '0';
+      j++;
+      if (ch < '0' || ch > '9')
+        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 = (FX_WORD)pTime->tm_year + 1900;
+    st.wMonth = (FX_WORD)pTime->tm_mon + 1;
+    st.wDay = (FX_WORD)pTime->tm_mday;
+    st.wDayOfWeek = (FX_WORD)pTime->tm_wday;
+    st.wHour = (FX_WORD)pTime->tm_hour;
+    st.wMinute = (FX_WORD)pTime->tm_min;
+    st.wSecond = (FX_WORD)pTime->tm_sec;
+    st.wMilliseconds = 0;
+  }
+}
+
+CPDFSDK_DateTime CPDFSDK_DateTime::ToGMT() {
+  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 (((FX_WORD)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 (((FX_WORD)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_Annot(CPDF_Annot* pAnnot, CPDFSDK_PageView* pPageView)
+    : m_pAnnot(pAnnot),
+      m_pPageView(pPageView),
+      m_bSelected(FALSE),
+      m_nTabOrder(-1) {}
+
+CPDFSDK_Annot::~CPDFSDK_Annot() {
+  m_pAnnot = NULL;
+  m_pPageView = NULL;
+}
+
+CPDF_Annot* CPDFSDK_Annot::GetPDFAnnot() {
+  return m_pAnnot;
+}
+
+FX_DWORD CPDFSDK_Annot::GetFlags() {
+  ASSERT(m_pAnnot != NULL);
+
+  return m_pAnnot->GetFlags();
+}
+
+void CPDFSDK_Annot::SetPage(CPDFSDK_PageView* pPageView) {
+  m_pPageView = pPageView;
+}
+
+CPDFSDK_PageView* CPDFSDK_Annot::GetPageView() {
+  return m_pPageView;
+}
+
+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_Annot::GetAnnotDict() const {
+  ASSERT(m_pAnnot != NULL);
+
+  return m_pAnnot->GetAnnotDict();
+}
+
+void CPDFSDK_Annot::SetRect(const CPDF_Rect& rect) {
+  ASSERT(rect.right - rect.left >= GetMinWidth());
+  ASSERT(rect.top - rect.bottom >= GetMinHeight());
+
+  m_pAnnot->GetAnnotDict()->SetAtRect("Rect", rect);
+}
+
+CPDF_Rect CPDFSDK_Annot::GetRect() const {
+  ASSERT(m_pAnnot != NULL);
+
+  CPDF_Rect rect;
+  m_pAnnot->GetRect(rect);
+
+  return rect;
+}
+
+CFX_ByteString CPDFSDK_Annot::GetType() const {
+  ASSERT(m_pAnnot != NULL);
+
+  return m_pAnnot->GetSubType();
+}
+
+CFX_ByteString CPDFSDK_Annot::GetSubType() const {
+  return "";
+}
+
+void CPDFSDK_Annot::DrawAppearance(CFX_RenderDevice* pDevice,
+                                   const CPDF_Matrix* pUser2Device,
+                                   CPDF_Annot::AppearanceMode mode,
+                                   const CPDF_RenderOptions* pOptions) {
+  ASSERT(m_pPageView != NULL);
+  ASSERT(m_pAnnot != NULL);
+
+  m_pAnnot->DrawAppearance(m_pPageView->GetPDFPage(), pDevice, pUser2Device,
+                           mode, pOptions);
+}
+
+FX_BOOL CPDFSDK_Annot::IsAppearanceValid() {
+  return m_pAnnot->GetAnnotDict()->GetDict("AP") != NULL;
+}
+
+FX_BOOL CPDFSDK_Annot::IsAppearanceValid(CPDF_Annot::AppearanceMode mode) {
+  CPDF_Dictionary* pAP = m_pAnnot->GetAnnotDict()->GetDict("AP");
+  if (pAP == NULL)
+    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->GetElementValue(ap_entry);
+  if (psub == NULL)
+    return FALSE;
+
+  return TRUE;
+}
+
+void CPDFSDK_Annot::DrawBorder(CFX_RenderDevice* pDevice,
+                               const CPDF_Matrix* pUser2Device,
+                               const CPDF_RenderOptions* pOptions) {
+  ASSERT(m_pAnnot != NULL);
+  m_pAnnot->DrawBorder(pDevice, pUser2Device, pOptions);
+}
+
+void CPDFSDK_Annot::ClearCachedAP() {
+  ASSERT(m_pAnnot != NULL);
+  m_pAnnot->ClearCachedAP();
+}
+
+void CPDFSDK_Annot::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_Annot::GetContents() const {
+  return m_pAnnot->GetAnnotDict()->GetUnicodeText("Contents");
+}
+
+void CPDFSDK_Annot::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_Annot::GetAnnotName() const {
+  return m_pAnnot->GetAnnotDict()->GetUnicodeText("NM");
+}
+
+void CPDFSDK_Annot::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_Annot::GetModifiedDate() const {
+  FX_SYSTEMTIME systime;
+  CFX_ByteString str = m_pAnnot->GetAnnotDict()->GetString("M");
+
+  CPDFSDK_DateTime dt(str);
+  dt.ToSystemTime(systime);
+
+  return systime;
+}
+
+void CPDFSDK_Annot::SetFlags(int nFlags) {
+  m_pAnnot->GetAnnotDict()->SetAtInteger("F", nFlags);
+}
+
+int CPDFSDK_Annot::GetFlags() const {
+  return m_pAnnot->GetAnnotDict()->GetInteger("F");
+}
+
+void CPDFSDK_Annot::SetAppState(const CFX_ByteString& str) {
+  if (str.IsEmpty())
+    m_pAnnot->GetAnnotDict()->RemoveAt("AS");
+  else
+    m_pAnnot->GetAnnotDict()->SetAtString("AS", str);
+}
+
+CFX_ByteString CPDFSDK_Annot::GetAppState() const {
+  return m_pAnnot->GetAnnotDict()->GetString("AS");
+}
+
+void CPDFSDK_Annot::SetStructParent(int key) {
+  m_pAnnot->GetAnnotDict()->SetAtInteger("StructParent", key);
+}
+
+int CPDFSDK_Annot::GetStructParent() const {
+  return m_pAnnot->GetAnnotDict()->GetInteger("StructParent");
+}
+
+// border
+void CPDFSDK_Annot::SetBorderWidth(int nWidth) {
+  CPDF_Array* pBorder = m_pAnnot->GetAnnotDict()->GetArray("Border");
+
+  if (pBorder) {
+    pBorder->SetAt(2, new CPDF_Number(nWidth));
+  } else {
+    CPDF_Dictionary* pBSDict = m_pAnnot->GetAnnotDict()->GetDict("BS");
+
+    if (!pBSDict) {
+      pBSDict = new CPDF_Dictionary;
+      m_pAnnot->GetAnnotDict()->SetAt("BS", pBSDict);
+    }
+
+    pBSDict->SetAtInteger("W", nWidth);
+  }
+}
+
+int CPDFSDK_Annot::GetBorderWidth() const {
+  if (CPDF_Array* pBorder = m_pAnnot->GetAnnotDict()->GetArray("Border")) {
+    return pBorder->GetInteger(2);
+  }
+  if (CPDF_Dictionary* pBSDict = m_pAnnot->GetAnnotDict()->GetDict("BS")) {
+    return pBSDict->GetInteger("W", 1);
+  }
+  return 1;
+}
+
+void CPDFSDK_Annot::SetBorderStyle(int nStyle) {
+  CPDF_Dictionary* pBSDict = m_pAnnot->GetAnnotDict()->GetDict("BS");
+  if (!pBSDict) {
+    pBSDict = new CPDF_Dictionary;
+    m_pAnnot->GetAnnotDict()->SetAt("BS", pBSDict);
+  }
+
+  switch (nStyle) {
+    case BBS_SOLID:
+      pBSDict->SetAtName("S", "S");
+      break;
+    case BBS_DASH:
+      pBSDict->SetAtName("S", "D");
+      break;
+    case BBS_BEVELED:
+      pBSDict->SetAtName("S", "B");
+      break;
+    case BBS_INSET:
+      pBSDict->SetAtName("S", "I");
+      break;
+    case BBS_UNDERLINE:
+      pBSDict->SetAtName("S", "U");
+      break;
+  }
+}
+
+int CPDFSDK_Annot::GetBorderStyle() const {
+  CPDF_Dictionary* pBSDict = m_pAnnot->GetAnnotDict()->GetDict("BS");
+  if (pBSDict) {
+    CFX_ByteString sBorderStyle = pBSDict->GetString("S", "S");
+    if (sBorderStyle == "S")
+      return BBS_SOLID;
+    if (sBorderStyle == "D")
+      return BBS_DASH;
+    if (sBorderStyle == "B")
+      return BBS_BEVELED;
+    if (sBorderStyle == "I")
+      return BBS_INSET;
+    if (sBorderStyle == "U")
+      return BBS_UNDERLINE;
+  }
+
+  CPDF_Array* pBorder = m_pAnnot->GetAnnotDict()->GetArray("Border");
+  if (pBorder) {
+    if (pBorder->GetCount() >= 4) {
+      CPDF_Array* pDP = pBorder->GetArray(3);
+      if (pDP && pDP->GetCount() > 0)
+        return BBS_DASH;
+    }
+  }
+
+  return BBS_SOLID;
+}
+
+void CPDFSDK_Annot::SetBorderDash(const CFX_IntArray& array) {
+  CPDF_Dictionary* pBSDict = m_pAnnot->GetAnnotDict()->GetDict("BS");
+  if (!pBSDict) {
+    pBSDict = new CPDF_Dictionary;
+    m_pAnnot->GetAnnotDict()->SetAt("BS", pBSDict);
+  }
+
+  CPDF_Array* pArray = new CPDF_Array;
+  for (int i = 0, sz = array.GetSize(); i < sz; i++) {
+    pArray->AddInteger(array[i]);
+  }
+
+  pBSDict->SetAt("D", pArray);
+}
+
+void CPDFSDK_Annot::GetBorderDash(CFX_IntArray& array) const {
+  CPDF_Array* pDash = NULL;
+
+  CPDF_Array* pBorder = m_pAnnot->GetAnnotDict()->GetArray("Border");
+  if (pBorder) {
+    pDash = pBorder->GetArray(3);
+  } else {
+    CPDF_Dictionary* pBSDict = m_pAnnot->GetAnnotDict()->GetDict("BS");
+    if (pBSDict) {
+      pDash = pBSDict->GetArray("D");
+    }
+  }
+
+  if (pDash) {
+    for (int i = 0, sz = pDash->GetCount(); i < sz; i++) {
+      array.Add(pDash->GetInteger(i));
+    }
+  }
+}
+
+void CPDFSDK_Annot::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_Annot::RemoveColor() {
+  m_pAnnot->GetAnnotDict()->RemoveAt("C");
+}
+
+FX_BOOL CPDFSDK_Annot::GetColor(FX_COLORREF& color) const {
+  if (CPDF_Array* pEntry = m_pAnnot->GetAnnotDict()->GetArray("C")) {
+    int nCount = pEntry->GetCount();
+    if (nCount == 1) {
+      FX_FLOAT g = pEntry->GetNumber(0) * 255;
+
+      color = FXSYS_RGB((int)g, (int)g, (int)g);
+
+      return TRUE;
+    } else if (nCount == 3) {
+      FX_FLOAT r = pEntry->GetNumber(0) * 255;
+      FX_FLOAT g = pEntry->GetNumber(1) * 255;
+      FX_FLOAT b = pEntry->GetNumber(2) * 255;
+
+      color = FXSYS_RGB((int)r, (int)g, (int)b);
+
+      return TRUE;
+    } else if (nCount == 4) {
+      FX_FLOAT c = pEntry->GetNumber(0);
+      FX_FLOAT m = pEntry->GetNumber(1);
+      FX_FLOAT y = pEntry->GetNumber(2);
+      FX_FLOAT k = pEntry->GetNumber(3);
+
+      FX_FLOAT r = 1.0f - FX_MIN(1.0f, c + k);
+      FX_FLOAT g = 1.0f - FX_MIN(1.0f, m + k);
+      FX_FLOAT b = 1.0f - FX_MIN(1.0f, y + k);
+
+      color = FXSYS_RGB((int)(r * 255), (int)(g * 255), (int)(b * 255));
+
+      return TRUE;
+    }
+  }
+
+  return FALSE;
+}
+
+void CPDFSDK_Annot::WriteAppearance(const CFX_ByteString& sAPType,
+                                    const CPDF_Rect& rcBBox,
+                                    const CPDF_Matrix& matrix,
+                                    const CFX_ByteString& sContents,
+                                    const CFX_ByteString& sAPState) {
+  CPDF_Dictionary* pAPDict = m_pAnnot->GetAnnotDict()->GetDict("AP");
+
+  if (!pAPDict) {
+    pAPDict = new CPDF_Dictionary;
+    m_pAnnot->GetAnnotDict()->SetAt("AP", pAPDict);
+  }
+
+  CPDF_Stream* pStream = NULL;
+  CPDF_Dictionary* pParentDict = NULL;
+
+  if (sAPState.IsEmpty()) {
+    pParentDict = pAPDict;
+    pStream = pAPDict->GetStream(sAPType);
+  } else {
+    CPDF_Dictionary* pAPTypeDict = pAPDict->GetDict(sAPType);
+    if (!pAPTypeDict) {
+      pAPTypeDict = new CPDF_Dictionary;
+      pAPDict->SetAt(sAPType, pAPTypeDict);
+    }
+
+    pParentDict = pAPTypeDict;
+    pStream = pAPTypeDict->GetStream(sAPState);
+  }
+
+  if (!pStream) {
+    CPDF_Document* pDoc = m_pPageView->GetPDFDocument();
+    pStream = new CPDF_Stream(NULL, 0, NULL);
+    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(NULL, 0, pStreamDict);
+  }
+
+  if (pStreamDict) {
+    pStreamDict->SetAtMatrix("Matrix", matrix);
+    pStreamDict->SetAtRect("BBox", rcBBox);
+  }
+
+  pStream->SetData((uint8_t*)sContents.c_str(), sContents.GetLength(), FALSE,
+                   FALSE);
+}
+
+#define BA_ANNOT_MINWIDTH 1
+#define BA_ANNOT_MINHEIGHT 1
+
+FX_FLOAT CPDFSDK_Annot::GetMinWidth() const {
+  return BA_ANNOT_MINWIDTH;
+}
+
+FX_FLOAT CPDFSDK_Annot::GetMinHeight() const {
+  return BA_ANNOT_MINHEIGHT;
+}
+
+FX_BOOL CPDFSDK_Annot::CreateFormFiller() {
+  return TRUE;
+}
+FX_BOOL CPDFSDK_Annot::IsVisible() const {
+  int nFlags = GetFlags();
+  return !((nFlags & ANNOTFLAG_INVISIBLE) || (nFlags & ANNOTFLAG_HIDDEN) ||
+           (nFlags & ANNOTFLAG_NOVIEW));
+}
+
+CPDF_Action CPDFSDK_Annot::GetAction() const {
+  return CPDF_Action(m_pAnnot->GetAnnotDict()->GetDict("A"));
+}
+
+void CPDFSDK_Annot::SetAction(const CPDF_Action& action) {
+  ASSERT(action);
+  if ((CPDF_Action&)action !=
+      CPDF_Action(m_pAnnot->GetAnnotDict()->GetDict("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_Annot::RemoveAction() {
+  m_pAnnot->GetAnnotDict()->RemoveAt("A");
+}
+
+CPDF_AAction CPDFSDK_Annot::GetAAction() const {
+  return m_pAnnot->GetAnnotDict()->GetDict("AA");
+}
+
+void CPDFSDK_Annot::SetAAction(const CPDF_AAction& aa) {
+  ASSERT(aa != NULL);
+
+  if ((CPDF_AAction&)aa != m_pAnnot->GetAnnotDict()->GetDict("AA"))
+    m_pAnnot->GetAnnotDict()->SetAt("AA", (CPDF_AAction&)aa);
+}
+
+void CPDFSDK_Annot::RemoveAAction() {
+  m_pAnnot->GetAnnotDict()->RemoveAt("AA");
+}
+
+CPDF_Action CPDFSDK_Annot::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_Annot::Annot_OnDraw(CFX_RenderDevice* pDevice,
+                                 CPDF_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);
+
+  return;
+}
+
+CPDF_Page* CPDFSDK_Annot::GetPDFPage() {
+  if (m_pPageView)
+    return m_pPageView->GetPDFPage();
+  return NULL;
+}