9cbb9de4d9a4ace17ddfb5e7fb80fefcd04ab003
[pdfium.git] / fpdfsdk / src / fsdk_mgr.cpp
1 // Copyright 2014 PDFium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4
5 // Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com
6
7 #include "../../public/fpdf_ext.h"
8 #include "../../third_party/base/nonstd_unique_ptr.h"
9 #include "../include/fsdk_define.h"
10 #include "../include/fsdk_mgr.h"
11 #include "../include/formfiller/FFL_FormFiller.h"
12 #include "../include/javascript/IJavaScript.h"
13
14 #if _FX_OS_ == _FX_ANDROID_
15 #include "time.h"
16 #else
17 #include <ctime>
18 #endif
19
20 class CFX_SystemHandler : public IFX_SystemHandler {
21  public:
22   CFX_SystemHandler(CPDFDoc_Environment* pEnv) : m_pEnv(pEnv), m_nCharSet(-1) {}
23   ~CFX_SystemHandler() override {}
24
25  public:
26   // IFX_SystemHandler
27   void InvalidateRect(FX_HWND hWnd, FX_RECT rect) override;
28   void OutputSelectedRect(void* pFormFiller, CPDF_Rect& rect) override;
29   FX_BOOL IsSelectionImplemented() override;
30   CFX_WideString GetClipboardText(FX_HWND hWnd) override { return L""; }
31   FX_BOOL SetClipboardText(FX_HWND hWnd, CFX_WideString string) override {
32     return FALSE;
33   }
34   void ClientToScreen(FX_HWND hWnd, int32_t& x, int32_t& y) override {}
35   void ScreenToClient(FX_HWND hWnd, int32_t& x, int32_t& y) override {}
36   void SetCursor(int32_t nCursorType) override;
37   FX_HMENU CreatePopupMenu() override { return NULL; }
38   FX_BOOL AppendMenuItem(FX_HMENU hMenu,
39                          int32_t nIDNewItem,
40                          CFX_WideString string) override {
41     return FALSE;
42   }
43   FX_BOOL EnableMenuItem(FX_HMENU hMenu,
44                          int32_t nIDItem,
45                          FX_BOOL bEnabled) override {
46     return FALSE;
47   }
48   int32_t TrackPopupMenu(FX_HMENU hMenu,
49                          int32_t x,
50                          int32_t y,
51                          FX_HWND hParent) override {
52     return -1;
53   }
54   void DestroyMenu(FX_HMENU hMenu) override {}
55   CFX_ByteString GetNativeTrueTypeFont(int32_t nCharset) override;
56   FX_BOOL FindNativeTrueTypeFont(int32_t nCharset,
57                                  CFX_ByteString sFontFaceName) override;
58   CPDF_Font* AddNativeTrueTypeFontToPDF(CPDF_Document* pDoc,
59                                         CFX_ByteString sFontFaceName,
60                                         uint8_t nCharset) override;
61   int32_t SetTimer(int32_t uElapse, TimerCallback lpTimerFunc) override;
62   void KillTimer(int32_t nID) override;
63   FX_BOOL IsSHIFTKeyDown(FX_DWORD nFlag) override {
64     return m_pEnv->FFI_IsSHIFTKeyDown(nFlag);
65   }
66   FX_BOOL IsCTRLKeyDown(FX_DWORD nFlag) override {
67     return m_pEnv->FFI_IsCTRLKeyDown(nFlag);
68   }
69   FX_BOOL IsALTKeyDown(FX_DWORD nFlag) override {
70     return m_pEnv->FFI_IsALTKeyDown(nFlag);
71   }
72   FX_BOOL IsINSERTKeyDown(FX_DWORD nFlag) override {
73     return m_pEnv->FFI_IsINSERTKeyDown(nFlag);
74   }
75   FX_SYSTEMTIME GetLocalTime() override;
76   int32_t GetCharSet() override { return m_nCharSet; }
77   void SetCharSet(int32_t nCharSet) override { m_nCharSet = nCharSet; }
78
79  private:
80   CPDFDoc_Environment* m_pEnv;
81   int m_nCharSet;
82 };
83
84 void CFX_SystemHandler::SetCursor(int32_t nCursorType) {
85   m_pEnv->FFI_SetCursor(nCursorType);
86 }
87
88 void CFX_SystemHandler::InvalidateRect(FX_HWND hWnd, FX_RECT rect) {
89   // g_pFormFillApp->FFI_Invalidate();
90   CPDFSDK_Annot* pSDKAnnot = (CPDFSDK_Annot*)hWnd;
91   CPDF_Page* pPage = NULL;
92   CPDFSDK_PageView* pPageView = NULL;
93   pPageView = pSDKAnnot->GetPageView();
94   pPage = pSDKAnnot->GetPDFPage();
95   if (!pPage || !pPageView)
96     return;
97   CPDF_Matrix page2device;
98   pPageView->GetCurrentMatrix(page2device);
99   CPDF_Matrix device2page;
100   device2page.SetReverse(page2device);
101   FX_FLOAT left, top, right, bottom;
102   device2page.Transform((FX_FLOAT)rect.left, (FX_FLOAT)rect.top, left, top);
103   device2page.Transform((FX_FLOAT)rect.right, (FX_FLOAT)rect.bottom, right,
104                         bottom);
105   //  m_pEnv->FFI_DeviceToPage(pPage, rect.left, rect.top, (double*)&left,
106   //  (double*)&top);
107   //  m_pEnv->FFI_DeviceToPage(pPage, rect.right, rect.bottom, (double*)&right,
108   //  (double*)&bottom);
109   CPDF_Rect rcPDF(left, bottom, right, top);
110   rcPDF.Normalize();
111
112   m_pEnv->FFI_Invalidate(pPage, rcPDF.left, rcPDF.top, rcPDF.right,
113                          rcPDF.bottom);
114 }
115 void CFX_SystemHandler::OutputSelectedRect(void* pFormFiller, CPDF_Rect& rect) {
116   CFFL_FormFiller* pFFL = (CFFL_FormFiller*)pFormFiller;
117   if (pFFL) {
118     CPDF_Point leftbottom = CPDF_Point(rect.left, rect.bottom);
119     CPDF_Point righttop = CPDF_Point(rect.right, rect.top);
120     CPDF_Point ptA = pFFL->PWLtoFFL(leftbottom);
121     CPDF_Point ptB = pFFL->PWLtoFFL(righttop);
122
123     CPDFSDK_Annot* pAnnot = pFFL->GetSDKAnnot();
124     ASSERT(pAnnot);
125     CPDF_Page* pPage = pAnnot->GetPDFPage();
126     ASSERT(pPage);
127     m_pEnv->FFI_OutputSelectedRect(pPage, ptA.x, ptB.y, ptB.x, ptA.y);
128   }
129 }
130
131 FX_BOOL CFX_SystemHandler::IsSelectionImplemented() {
132   if (m_pEnv) {
133     FPDF_FORMFILLINFO* pInfo = m_pEnv->GetFormFillInfo();
134     if (pInfo && pInfo->FFI_OutputSelectedRect)
135       return TRUE;
136   }
137   return FALSE;
138 }
139
140 CFX_ByteString CFX_SystemHandler::GetNativeTrueTypeFont(int32_t nCharset) {
141   return "";
142 }
143
144 FX_BOOL CFX_SystemHandler::FindNativeTrueTypeFont(
145     int32_t nCharset,
146     CFX_ByteString sFontFaceName) {
147   CFX_FontMgr* pFontMgr = CFX_GEModule::Get()->GetFontMgr();
148   if (pFontMgr) {
149     CFX_FontMapper* pFontMapper = pFontMgr->m_pBuiltinMapper;
150     if (pFontMapper) {
151       int nSize = pFontMapper->m_InstalledTTFonts.GetSize();
152       if (nSize == 0) {
153         pFontMapper->LoadInstalledFonts();
154         nSize = pFontMapper->m_InstalledTTFonts.GetSize();
155       }
156
157       for (int i = 0; i < nSize; i++) {
158         if (pFontMapper->m_InstalledTTFonts[i].Compare(sFontFaceName))
159           return TRUE;
160       }
161     }
162   }
163
164   return FALSE;
165 }
166
167 static int CharSet2CP(int charset) {
168   if (charset == 128)
169     return 932;
170   if (charset == 134)
171     return 936;
172   if (charset == 129)
173     return 949;
174   if (charset == 136)
175     return 950;
176   return 0;
177 }
178 CPDF_Font* CFX_SystemHandler::AddNativeTrueTypeFontToPDF(
179     CPDF_Document* pDoc,
180     CFX_ByteString sFontFaceName,
181     uint8_t nCharset) {
182   if (pDoc) {
183     CFX_Font* pFXFont = new CFX_Font();
184     pFXFont->LoadSubst(sFontFaceName, TRUE, 0, 0, 0, CharSet2CP(nCharset),
185                        FALSE);
186     CPDF_Font* pFont = pDoc->AddFont(pFXFont, nCharset, FALSE);
187     delete pFXFont;
188     return pFont;
189   }
190
191   return NULL;
192 }
193
194 int32_t CFX_SystemHandler::SetTimer(int32_t uElapse,
195                                     TimerCallback lpTimerFunc) {
196   return m_pEnv->FFI_SetTimer(uElapse, lpTimerFunc);
197 }
198 void CFX_SystemHandler::KillTimer(int32_t nID) {
199   m_pEnv->FFI_KillTimer(nID);
200 }
201
202 FX_SYSTEMTIME CFX_SystemHandler::GetLocalTime() {
203   return m_pEnv->FFI_GetLocalTime();
204 }
205
206 CJS_RuntimeFactory* GetJSRuntimeFactory() {
207   static CJS_RuntimeFactory s_JSRuntimeFactory;
208   return &s_JSRuntimeFactory;
209 }
210
211 CPDFDoc_Environment::CPDFDoc_Environment(CPDF_Document* pDoc,
212                                          FPDF_FORMFILLINFO* pFFinfo)
213     : m_pAnnotHandlerMgr(NULL),
214       m_pActionHandler(NULL),
215       m_pJSRuntime(NULL),
216       m_pInfo(pFFinfo),
217       m_pSDKDoc(NULL),
218       m_pPDFDoc(pDoc),
219       m_pIFormFiller(NULL) {
220   m_pSysHandler = new CFX_SystemHandler(this);
221   m_pJSRuntimeFactory = GetJSRuntimeFactory();
222   m_pJSRuntimeFactory->AddRef();
223 }
224
225 CPDFDoc_Environment::~CPDFDoc_Environment() {
226   delete m_pIFormFiller;
227   m_pIFormFiller = NULL;
228   if (m_pJSRuntime && m_pJSRuntimeFactory)
229     m_pJSRuntimeFactory->DeleteJSRuntime(m_pJSRuntime);
230   m_pJSRuntimeFactory->Release();
231
232   delete m_pSysHandler;
233   m_pSysHandler = NULL;
234
235   delete m_pAnnotHandlerMgr;
236   m_pAnnotHandlerMgr = NULL;
237   delete m_pActionHandler;
238   m_pActionHandler = NULL;
239 }
240
241 int CPDFDoc_Environment::JS_appAlert(const FX_WCHAR* Msg,
242                                      const FX_WCHAR* Title,
243                                      FX_UINT Type,
244                                      FX_UINT Icon) {
245   if (m_pInfo && m_pInfo->m_pJsPlatform && m_pInfo->m_pJsPlatform->app_alert) {
246     CFX_ByteString bsMsg = CFX_WideString(Msg).UTF16LE_Encode();
247     CFX_ByteString bsTitle = CFX_WideString(Title).UTF16LE_Encode();
248     FPDF_WIDESTRING pMsg = (FPDF_WIDESTRING)bsMsg.GetBuffer(bsMsg.GetLength());
249     FPDF_WIDESTRING pTitle =
250         (FPDF_WIDESTRING)bsTitle.GetBuffer(bsTitle.GetLength());
251     int ret = m_pInfo->m_pJsPlatform->app_alert(m_pInfo->m_pJsPlatform, pMsg,
252                                                 pTitle, Type, Icon);
253     bsMsg.ReleaseBuffer();
254     bsTitle.ReleaseBuffer();
255     return ret;
256   }
257   return -1;
258 }
259
260 int CPDFDoc_Environment::JS_appResponse(const FX_WCHAR* Question,
261                                         const FX_WCHAR* Title,
262                                         const FX_WCHAR* Default,
263                                         const FX_WCHAR* cLabel,
264                                         FPDF_BOOL bPassword,
265                                         void* response,
266                                         int length) {
267   if (m_pInfo && m_pInfo->m_pJsPlatform &&
268       m_pInfo->m_pJsPlatform->app_response) {
269     CFX_ByteString bsQuestion = CFX_WideString(Question).UTF16LE_Encode();
270     CFX_ByteString bsTitle = CFX_WideString(Title).UTF16LE_Encode();
271     CFX_ByteString bsDefault = CFX_WideString(Default).UTF16LE_Encode();
272     CFX_ByteString bsLabel = CFX_WideString(cLabel).UTF16LE_Encode();
273     FPDF_WIDESTRING pQuestion =
274         (FPDF_WIDESTRING)bsQuestion.GetBuffer(bsQuestion.GetLength());
275     FPDF_WIDESTRING pTitle =
276         (FPDF_WIDESTRING)bsTitle.GetBuffer(bsTitle.GetLength());
277     FPDF_WIDESTRING pDefault =
278         (FPDF_WIDESTRING)bsDefault.GetBuffer(bsDefault.GetLength());
279     FPDF_WIDESTRING pLabel =
280         (FPDF_WIDESTRING)bsLabel.GetBuffer(bsLabel.GetLength());
281     int ret = m_pInfo->m_pJsPlatform->app_response(
282         m_pInfo->m_pJsPlatform, pQuestion, pTitle, pDefault, pLabel, bPassword,
283         response, length);
284     bsQuestion.ReleaseBuffer();
285     bsTitle.ReleaseBuffer();
286     bsDefault.ReleaseBuffer();
287     bsLabel.ReleaseBuffer();
288     return ret;
289   }
290   return -1;
291 }
292
293 CFX_WideString CPDFDoc_Environment::JS_fieldBrowse() {
294   if (!m_pInfo || !m_pInfo->m_pJsPlatform ||
295       !m_pInfo->m_pJsPlatform->Field_browse) {
296     return L"";
297   }
298
299   const int nRequiredLen =
300       m_pInfo->m_pJsPlatform->Field_browse(m_pInfo->m_pJsPlatform, nullptr, 0);
301   if (nRequiredLen <= 0)
302     return L"";
303
304   nonstd::unique_ptr<char[]> pBuff(new char[nRequiredLen]);
305   memset(pBuff.get(), 0, nRequiredLen);
306   const int nActualLen = m_pInfo->m_pJsPlatform->Field_browse(
307       m_pInfo->m_pJsPlatform, pBuff.get(), nRequiredLen);
308   if (nActualLen <= 0 || nActualLen > nRequiredLen)
309     return L"";
310
311   CFX_ByteString bsRet = CFX_ByteString(pBuff.get(), nActualLen);
312   CFX_WideString wsRet = CFX_WideString::FromLocal(bsRet);
313   return wsRet;
314 }
315
316 CFX_WideString CPDFDoc_Environment::JS_docGetFilePath() {
317   if (!m_pInfo || !m_pInfo->m_pJsPlatform ||
318       !m_pInfo->m_pJsPlatform->Doc_getFilePath) {
319     return L"";
320   }
321
322   const int nRequiredLen = m_pInfo->m_pJsPlatform->Doc_getFilePath(
323       m_pInfo->m_pJsPlatform, nullptr, 0);
324   if (nRequiredLen <= 0)
325     return L"";
326
327   nonstd::unique_ptr<char[]> pBuff(new char[nRequiredLen]);
328   memset(pBuff.get(), 0, nRequiredLen);
329   const int nActualLen = m_pInfo->m_pJsPlatform->Doc_getFilePath(
330       m_pInfo->m_pJsPlatform, pBuff.get(), nRequiredLen);
331   if (nActualLen <= 0 || nActualLen > nRequiredLen)
332     return L"";
333
334   CFX_ByteString bsRet = CFX_ByteString(pBuff.get(), nActualLen);
335   CFX_WideString wsRet = CFX_WideString::FromLocal(bsRet);
336   return wsRet;
337 }
338
339 void CPDFDoc_Environment::JS_docSubmitForm(void* formData,
340                                            int length,
341                                            const FX_WCHAR* URL) {
342   if (m_pInfo && m_pInfo->m_pJsPlatform &&
343       m_pInfo->m_pJsPlatform->Doc_submitForm) {
344     CFX_ByteString bsDestination = CFX_WideString(URL).UTF16LE_Encode();
345     FPDF_WIDESTRING pDestination =
346         (FPDF_WIDESTRING)bsDestination.GetBuffer(bsDestination.GetLength());
347     m_pInfo->m_pJsPlatform->Doc_submitForm(m_pInfo->m_pJsPlatform, formData,
348                                            length, pDestination);
349     bsDestination.ReleaseBuffer();
350   }
351 }
352
353 void CPDFDoc_Environment::JS_docmailForm(void* mailData,
354                                          int length,
355                                          FPDF_BOOL bUI,
356                                          const FX_WCHAR* To,
357                                          const FX_WCHAR* Subject,
358                                          const FX_WCHAR* CC,
359                                          const FX_WCHAR* BCC,
360                                          const FX_WCHAR* Msg) {
361   if (m_pInfo && m_pInfo->m_pJsPlatform && m_pInfo->m_pJsPlatform->Doc_mail) {
362     CFX_ByteString bsTo = CFX_WideString(To).UTF16LE_Encode();
363     CFX_ByteString bsCC = CFX_WideString(Subject).UTF16LE_Encode();
364     CFX_ByteString bsBcc = CFX_WideString(BCC).UTF16LE_Encode();
365     CFX_ByteString bsSubject = CFX_WideString(Subject).UTF16LE_Encode();
366     CFX_ByteString bsMsg = CFX_WideString(Msg).UTF16LE_Encode();
367     FPDF_WIDESTRING pTo = (FPDF_WIDESTRING)bsTo.GetBuffer(bsTo.GetLength());
368     FPDF_WIDESTRING pCC = (FPDF_WIDESTRING)bsCC.GetBuffer(bsCC.GetLength());
369     FPDF_WIDESTRING pBcc = (FPDF_WIDESTRING)bsBcc.GetBuffer(bsBcc.GetLength());
370     FPDF_WIDESTRING pSubject =
371         (FPDF_WIDESTRING)bsSubject.GetBuffer(bsSubject.GetLength());
372     FPDF_WIDESTRING pMsg = (FPDF_WIDESTRING)bsMsg.GetBuffer(bsMsg.GetLength());
373     m_pInfo->m_pJsPlatform->Doc_mail(m_pInfo->m_pJsPlatform, mailData, length,
374                                      bUI, pTo, pSubject, pCC, pBcc, pMsg);
375     bsTo.ReleaseBuffer();
376     bsCC.ReleaseBuffer();
377     bsBcc.ReleaseBuffer();
378     bsSubject.ReleaseBuffer();
379     bsMsg.ReleaseBuffer();
380   }
381 }
382
383 IFXJS_Runtime* CPDFDoc_Environment::GetJSRuntime() {
384   if (!IsJSInitiated())
385     return NULL;
386   if (!m_pJSRuntime)
387     m_pJSRuntime = m_pJSRuntimeFactory->NewJSRuntime(this);
388   return m_pJSRuntime;
389 }
390
391 CPDFSDK_AnnotHandlerMgr* CPDFDoc_Environment::GetAnnotHandlerMgr() {
392   if (!m_pAnnotHandlerMgr)
393     m_pAnnotHandlerMgr = new CPDFSDK_AnnotHandlerMgr(this);
394   return m_pAnnotHandlerMgr;
395 }
396
397 CPDFSDK_ActionHandler* CPDFDoc_Environment::GetActionHander() {
398   if (!m_pActionHandler)
399     m_pActionHandler = new CPDFSDK_ActionHandler();
400   return m_pActionHandler;
401 }
402
403 CFFL_IFormFiller* CPDFDoc_Environment::GetIFormFiller() {
404   if (!m_pIFormFiller)
405     m_pIFormFiller = new CFFL_IFormFiller(this);
406   return m_pIFormFiller;
407 }
408
409 CPDFSDK_Document::CPDFSDK_Document(CPDF_Document* pDoc,
410                                    CPDFDoc_Environment* pEnv)
411     : m_pDoc(pDoc),
412       m_pInterForm(nullptr),
413       m_pFocusAnnot(nullptr),
414       m_pEnv(pEnv),
415       m_pOccontent(nullptr),
416       m_bChangeMask(FALSE) {}
417
418 CPDFSDK_Document::~CPDFSDK_Document() {
419   for (auto& it : m_pageMap)
420     delete it.second;
421   m_pageMap.clear();
422
423   delete m_pInterForm;
424   m_pInterForm = nullptr;
425
426   delete m_pOccontent;
427   m_pOccontent = nullptr;
428 }
429
430 CPDFSDK_PageView* CPDFSDK_Document::GetPageView(CPDF_Page* pPDFPage,
431                                                 FX_BOOL ReNew) {
432   auto it = m_pageMap.find(pPDFPage);
433   if (it != m_pageMap.end())
434     return it->second;
435
436   if (!ReNew)
437     return nullptr;
438
439   CPDFSDK_PageView* pPageView = new CPDFSDK_PageView(this, pPDFPage);
440   m_pageMap[pPDFPage] = pPageView;
441   // Delay to load all the annotations, to avoid endless loop.
442   pPageView->LoadFXAnnots();
443   return pPageView;
444 }
445
446 CPDFSDK_PageView* CPDFSDK_Document::GetCurrentView() {
447   CPDF_Page* pPage = (CPDF_Page*)m_pEnv->FFI_GetCurrentPage(m_pDoc);
448   return pPage ? GetPageView(pPage, TRUE) : nullptr;
449 }
450
451 CPDFSDK_PageView* CPDFSDK_Document::GetPageView(int nIndex) {
452   CPDF_Page* pTempPage = (CPDF_Page*)m_pEnv->FFI_GetPage(m_pDoc, nIndex);
453   if (!pTempPage)
454     return nullptr;
455
456   auto it = m_pageMap.find(pTempPage);
457   return it->second;
458 }
459
460 void CPDFSDK_Document::ProcJavascriptFun() {
461   CPDF_Document* pPDFDoc = GetDocument();
462   CPDF_DocJSActions docJS(pPDFDoc);
463   int iCount = docJS.CountJSActions();
464   if (iCount < 1)
465     return;
466   for (int i = 0; i < iCount; i++) {
467     CFX_ByteString csJSName;
468     CPDF_Action jsAction = docJS.GetJSAction(i, csJSName);
469     if (m_pEnv->GetActionHander())
470       m_pEnv->GetActionHander()->DoAction_JavaScript(
471           jsAction, CFX_WideString::FromLocal(csJSName), this);
472   }
473 }
474
475 FX_BOOL CPDFSDK_Document::ProcOpenAction() {
476   if (!m_pDoc)
477     return FALSE;
478
479   CPDF_Dictionary* pRoot = m_pDoc->GetRoot();
480   if (!pRoot)
481     return FALSE;
482
483   CPDF_Object* pOpenAction = pRoot->GetDict("OpenAction");
484   if (!pOpenAction)
485     pOpenAction = pRoot->GetArray("OpenAction");
486
487   if (!pOpenAction)
488     return FALSE;
489
490   if (pOpenAction->GetType() == PDFOBJ_ARRAY)
491     return TRUE;
492
493   if (pOpenAction->GetType() == PDFOBJ_DICTIONARY) {
494     CPDF_Dictionary* pDict = (CPDF_Dictionary*)pOpenAction;
495     CPDF_Action action(pDict);
496     if (m_pEnv->GetActionHander())
497       m_pEnv->GetActionHander()->DoAction_DocOpen(action, this);
498     return TRUE;
499   }
500   return FALSE;
501 }
502
503 CPDF_OCContext* CPDFSDK_Document::GetOCContext() {
504   if (!m_pOccontent)
505     m_pOccontent = new CPDF_OCContext(m_pDoc);
506   return m_pOccontent;
507 }
508
509 void CPDFSDK_Document::ReMovePageView(CPDF_Page* pPDFPage) {
510   auto it = m_pageMap.find(pPDFPage);
511   if (it == m_pageMap.end())
512     return;
513
514   CPDFSDK_PageView* pPageView = it->second;
515   if (pPageView->IsLocked())
516     return;
517
518   delete pPageView;
519   m_pageMap.erase(it);
520 }
521
522 CPDF_Page* CPDFSDK_Document::GetPage(int nIndex) {
523   CPDF_Page* pTempPage = (CPDF_Page*)m_pEnv->FFI_GetPage(m_pDoc, nIndex);
524   if (!pTempPage)
525     return NULL;
526   return pTempPage;
527 }
528
529 CPDFSDK_InterForm* CPDFSDK_Document::GetInterForm() {
530   if (!m_pInterForm)
531     m_pInterForm = new CPDFSDK_InterForm(this);
532   return m_pInterForm;
533 }
534
535 void CPDFSDK_Document::UpdateAllViews(CPDFSDK_PageView* pSender,
536                                       CPDFSDK_Annot* pAnnot) {
537   for (const auto& it : m_pageMap) {
538     CPDFSDK_PageView* pPageView = it.second;
539     if (pPageView != pSender) {
540       pPageView->UpdateView(pAnnot);
541     }
542   }
543 }
544
545 CPDFSDK_Annot* CPDFSDK_Document::GetFocusAnnot() {
546   return m_pFocusAnnot;
547 }
548
549 FX_BOOL CPDFSDK_Document::SetFocusAnnot(CPDFSDK_Annot* pAnnot, FX_UINT nFlag) {
550   if (m_pFocusAnnot == pAnnot)
551     return TRUE;
552
553   if (m_pFocusAnnot) {
554     if (!KillFocusAnnot(nFlag))
555       return FALSE;
556   }
557   CPDFSDK_PageView* pPageView = pAnnot->GetPageView();
558   if (pAnnot && pPageView->IsValid()) {
559     CPDFSDK_AnnotHandlerMgr* pAnnotHandler = m_pEnv->GetAnnotHandlerMgr();
560
561     if (pAnnotHandler && !m_pFocusAnnot) {
562       if (!pAnnotHandler->Annot_OnSetFocus(pAnnot, nFlag))
563         return FALSE;
564       if (!m_pFocusAnnot) {
565         m_pFocusAnnot = pAnnot;
566         return TRUE;
567       }
568     }
569   }
570   return FALSE;
571 }
572
573 FX_BOOL CPDFSDK_Document::KillFocusAnnot(FX_UINT nFlag) {
574   if (m_pFocusAnnot) {
575     CPDFSDK_AnnotHandlerMgr* pAnnotHandler = m_pEnv->GetAnnotHandlerMgr();
576     if (pAnnotHandler) {
577       CPDFSDK_Annot* pFocusAnnot = m_pFocusAnnot;
578       m_pFocusAnnot = NULL;
579       if (pAnnotHandler->Annot_OnKillFocus(pFocusAnnot, nFlag)) {
580         if (pFocusAnnot->GetType() == FX_BSTRC("Widget")) {
581           CPDFSDK_Widget* pWidget = (CPDFSDK_Widget*)pFocusAnnot;
582           int nFieldType = pWidget->GetFieldType();
583           if (FIELDTYPE_TEXTFIELD == nFieldType ||
584               FIELDTYPE_COMBOBOX == nFieldType)
585             m_pEnv->FFI_OnSetFieldInputFocus(NULL, NULL, 0, FALSE);
586         }
587
588         if (!m_pFocusAnnot)
589           return TRUE;
590       } else {
591         m_pFocusAnnot = pFocusAnnot;
592       }
593     }
594   }
595   return FALSE;
596 }
597
598 void CPDFSDK_Document::OnCloseDocument() {
599   KillFocusAnnot();
600 }
601
602 FX_BOOL CPDFSDK_Document::GetPermissions(int nFlag) {
603   FX_DWORD dwPermissions = m_pDoc->GetUserPermissions();
604   return dwPermissions & nFlag;
605 }
606
607 IFXJS_Runtime* CPDFSDK_Document::GetJsRuntime() {
608   ASSERT(m_pEnv != NULL);
609   return m_pEnv->GetJSRuntime();
610 }
611
612 CFX_WideString CPDFSDK_Document::GetPath() {
613   ASSERT(m_pEnv != NULL);
614   return m_pEnv->JS_docGetFilePath();
615 }
616
617 CPDFSDK_PageView::CPDFSDK_PageView(CPDFSDK_Document* pSDKDoc, CPDF_Page* page)
618     : m_page(page), m_pSDKDoc(pSDKDoc) {
619   CPDFSDK_InterForm* pInterForm = pSDKDoc->GetInterForm();
620   if (pInterForm) {
621     CPDF_InterForm* pPDFInterForm = pInterForm->GetInterForm();
622     pPDFInterForm->FixPageFields(page);
623   }
624   m_page->SetPrivateData((void*)m_page, (void*)this, NULL);
625   m_fxAnnotArray.RemoveAll();
626
627   m_bEnterWidget = FALSE;
628   m_bExitWidget = FALSE;
629   m_bOnWidget = FALSE;
630   m_CaptureWidget = NULL;
631   m_bValid = FALSE;
632   m_bLocked = FALSE;
633   m_bTakeOverPage = FALSE;
634 }
635
636 CPDFSDK_PageView::~CPDFSDK_PageView() {
637   // if there is a focused annot on the page, we should kill the focus first.
638   if (CPDFSDK_Annot* focusedAnnot = m_pSDKDoc->GetFocusAnnot()) {
639     for (int i = 0, count = m_fxAnnotArray.GetSize(); i < count; i++) {
640       CPDFSDK_Annot* pAnnot = (CPDFSDK_Annot*)m_fxAnnotArray.GetAt(i);
641       if (pAnnot == focusedAnnot) {
642         KillFocusAnnot();
643         break;
644       }
645     }
646   }
647
648   CPDFDoc_Environment* pEnv = m_pSDKDoc->GetEnv();
649   CPDFSDK_AnnotHandlerMgr* pAnnotHandlerMgr = pEnv->GetAnnotHandlerMgr();
650   ASSERT(pAnnotHandlerMgr);
651   for (int i = 0, count = m_fxAnnotArray.GetSize(); i < count; i++) {
652     CPDFSDK_Annot* pAnnot = (CPDFSDK_Annot*)m_fxAnnotArray.GetAt(i);
653     pAnnotHandlerMgr->ReleaseAnnot(pAnnot);
654   }
655   m_fxAnnotArray.RemoveAll();
656
657   delete m_pAnnotList;
658   m_pAnnotList = NULL;
659
660   m_page->RemovePrivateData((void*)m_page);
661   if (m_bTakeOverPage) {
662     delete m_page;
663   }
664 }
665
666 void CPDFSDK_PageView::PageView_OnDraw(CFX_RenderDevice* pDevice,
667                                        CPDF_Matrix* pUser2Device,
668                                        CPDF_RenderOptions* pOptions) {
669   m_curMatrix = *pUser2Device;
670
671   CPDFDoc_Environment* pEnv = m_pSDKDoc->GetEnv();
672   CPDFSDK_AnnotIterator annotIterator(this, TRUE);
673   CPDFSDK_Annot* pSDKAnnot = nullptr;
674   int index = -1;
675   while ((pSDKAnnot = annotIterator.Next(index))) {
676     CPDFSDK_AnnotHandlerMgr* pAnnotHandlerMgr = pEnv->GetAnnotHandlerMgr();
677     ASSERT(pAnnotHandlerMgr);
678     pAnnotHandlerMgr->Annot_OnDraw(this, pSDKAnnot, pDevice, pUser2Device, 0);
679   }
680 }
681
682 CPDF_Annot* CPDFSDK_PageView::GetPDFAnnotAtPoint(FX_FLOAT pageX,
683                                                  FX_FLOAT pageY) {
684   int nCount = m_pAnnotList->Count();
685   for (int i = 0; i < nCount; i++) {
686     CPDF_Annot* pAnnot = m_pAnnotList->GetAt(i);
687     CFX_FloatRect annotRect;
688     pAnnot->GetRect(annotRect);
689     if (annotRect.Contains(pageX, pageY))
690       return pAnnot;
691   }
692   return NULL;
693 }
694
695 CPDF_Annot* CPDFSDK_PageView::GetPDFWidgetAtPoint(FX_FLOAT pageX,
696                                                   FX_FLOAT pageY) {
697   int nCount = m_pAnnotList->Count();
698   for (int i = 0; i < nCount; i++) {
699     CPDF_Annot* pAnnot = m_pAnnotList->GetAt(i);
700     if (pAnnot->GetSubType() == "Widget") {
701       CFX_FloatRect annotRect;
702       pAnnot->GetRect(annotRect);
703       if (annotRect.Contains(pageX, pageY))
704         return pAnnot;
705     }
706   }
707   return NULL;
708 }
709
710 CPDFSDK_Annot* CPDFSDK_PageView::GetFXAnnotAtPoint(FX_FLOAT pageX,
711                                                    FX_FLOAT pageY) {
712   CPDFSDK_AnnotIterator annotIterator(this, FALSE);
713   CPDFDoc_Environment* pEnv = m_pSDKDoc->GetEnv();
714   CPDFSDK_AnnotHandlerMgr* pAnnotMgr = pEnv->GetAnnotHandlerMgr();
715   CPDFSDK_Annot* pSDKAnnot = NULL;
716   int index = -1;
717   while ((pSDKAnnot = annotIterator.Next(index))) {
718     CPDF_Rect rc = pAnnotMgr->Annot_OnGetViewBBox(this, pSDKAnnot);
719     if (rc.Contains(pageX, pageY))
720       return pSDKAnnot;
721   }
722
723   return NULL;
724 }
725
726 CPDFSDK_Annot* CPDFSDK_PageView::GetFXWidgetAtPoint(FX_FLOAT pageX,
727                                                     FX_FLOAT pageY) {
728   CPDFSDK_AnnotIterator annotIterator(this, FALSE);
729   CPDFDoc_Environment* pEnv = m_pSDKDoc->GetEnv();
730   CPDFSDK_AnnotHandlerMgr* pAnnotMgr = pEnv->GetAnnotHandlerMgr();
731   CPDFSDK_Annot* pSDKAnnot = NULL;
732   int index = -1;
733   while ((pSDKAnnot = annotIterator.Next(index))) {
734     if (pSDKAnnot->GetType() == "Widget") {
735       pAnnotMgr->Annot_OnGetViewBBox(this, pSDKAnnot);
736       CPDF_Point point(pageX, pageY);
737       if (pAnnotMgr->Annot_OnHitTest(this, pSDKAnnot, point))
738         return pSDKAnnot;
739     }
740   }
741
742   return NULL;
743 }
744
745 FX_BOOL CPDFSDK_PageView::Annot_HasAppearance(CPDF_Annot* pAnnot) {
746   CPDF_Dictionary* pAnnotDic = pAnnot->GetAnnotDict();
747   if (pAnnotDic)
748     return pAnnotDic->KeyExist("AS");
749   return FALSE;
750 }
751
752 CPDFSDK_Annot* CPDFSDK_PageView::AddAnnot(CPDF_Annot* pPDFAnnot) {
753   CPDFDoc_Environment* pEnv = m_pSDKDoc->GetEnv();
754   ASSERT(pEnv);
755   CPDFSDK_AnnotHandlerMgr* pAnnotHandler = pEnv->GetAnnotHandlerMgr();
756
757   CPDFSDK_Annot* pSDKAnnot = NULL;
758
759   if (pAnnotHandler) {
760     pSDKAnnot = pAnnotHandler->NewAnnot(pPDFAnnot, this);
761   }
762   if (!pSDKAnnot)
763     return NULL;
764
765   m_fxAnnotArray.Add(pSDKAnnot);
766
767   if (pAnnotHandler) {
768     pAnnotHandler->Annot_OnCreate(pSDKAnnot);
769   }
770
771   return pSDKAnnot;
772 }
773
774 CPDFSDK_Annot* CPDFSDK_PageView::AddAnnot(CPDF_Dictionary* pDict) {
775   return pDict ? AddAnnot(pDict->GetString("Subtype"), pDict) : nullptr;
776 }
777
778 CPDFSDK_Annot* CPDFSDK_PageView::AddAnnot(const FX_CHAR* lpSubType,
779                                           CPDF_Dictionary* pDict) {
780   return NULL;
781 }
782
783 FX_BOOL CPDFSDK_PageView::DeleteAnnot(CPDFSDK_Annot* pAnnot) {
784   return FALSE;
785 }
786
787 CPDF_Document* CPDFSDK_PageView::GetPDFDocument() {
788   if (m_page) {
789     return m_page->m_pDocument;
790   }
791   return NULL;
792 }
793
794 int CPDFSDK_PageView::CountAnnots() {
795   return m_pAnnotList->Count();
796 }
797
798 CPDFSDK_Annot* CPDFSDK_PageView::GetAnnot(int nIndex) {
799   int nCount = m_fxAnnotArray.GetSize();
800   if (nIndex < 0 || nIndex >= nCount) {
801     return NULL;
802   }
803
804   return (CPDFSDK_Annot*)m_fxAnnotArray.GetAt(nIndex);
805 }
806
807 CPDFSDK_Annot* CPDFSDK_PageView::GetAnnotByDict(CPDF_Dictionary* pDict) {
808   int nCount = m_fxAnnotArray.GetSize();
809   for (int i = 0; i < nCount; i++) {
810     CPDFSDK_Annot* pAnnot = (CPDFSDK_Annot*)m_fxAnnotArray.GetAt(i);
811     if (pDict == pAnnot->GetPDFAnnot()->GetAnnotDict())
812       return pAnnot;
813   }
814   return NULL;
815 }
816
817 FX_BOOL CPDFSDK_PageView::OnLButtonDown(const CPDF_Point& point,
818                                         FX_UINT nFlag) {
819   CPDFDoc_Environment* pEnv = m_pSDKDoc->GetEnv();
820   ASSERT(pEnv);
821   CPDFSDK_Annot* pFXAnnot = GetFXWidgetAtPoint(point.x, point.y);
822   if (!pFXAnnot) {
823     KillFocusAnnot(nFlag);
824   } else {
825     CPDFSDK_AnnotHandlerMgr* pAnnotHandlerMgr = pEnv->GetAnnotHandlerMgr();
826     ASSERT(pAnnotHandlerMgr);
827
828     FX_BOOL bRet =
829         pAnnotHandlerMgr->Annot_OnLButtonDown(this, pFXAnnot, nFlag, point);
830     if (bRet) {
831       SetFocusAnnot(pFXAnnot);
832     }
833     return bRet;
834   }
835   return FALSE;
836 }
837
838 FX_BOOL CPDFSDK_PageView::OnLButtonUp(const CPDF_Point& point, FX_UINT nFlag) {
839   CPDFDoc_Environment* pEnv = m_pSDKDoc->GetEnv();
840   ASSERT(pEnv);
841   CPDFSDK_AnnotHandlerMgr* pAnnotHandlerMgr = pEnv->GetAnnotHandlerMgr();
842   ASSERT(pAnnotHandlerMgr);
843   CPDFSDK_Annot* pFXAnnot = GetFXWidgetAtPoint(point.x, point.y);
844   CPDFSDK_Annot* pFocusAnnot = GetFocusAnnot();
845   FX_BOOL bRet = FALSE;
846   if (pFocusAnnot && pFocusAnnot != pFXAnnot) {
847     // Last focus Annot gets a chance to handle the event.
848     bRet = pAnnotHandlerMgr->Annot_OnLButtonUp(this, pFocusAnnot, nFlag, point);
849   }
850   if (pFXAnnot && !bRet) {
851     bRet = pAnnotHandlerMgr->Annot_OnLButtonUp(this, pFXAnnot, nFlag, point);
852     return bRet;
853   }
854   return bRet;
855 }
856
857 FX_BOOL CPDFSDK_PageView::OnMouseMove(const CPDF_Point& point, int nFlag) {
858   CPDFDoc_Environment* pEnv = m_pSDKDoc->GetEnv();
859   CPDFSDK_AnnotHandlerMgr* pAnnotHandlerMgr = pEnv->GetAnnotHandlerMgr();
860   ASSERT(pAnnotHandlerMgr);
861   if (CPDFSDK_Annot* pFXAnnot = GetFXWidgetAtPoint(point.x, point.y)) {
862     if (m_CaptureWidget && m_CaptureWidget != pFXAnnot) {
863       m_bExitWidget = TRUE;
864       m_bEnterWidget = FALSE;
865       pAnnotHandlerMgr->Annot_OnMouseExit(this, m_CaptureWidget, nFlag);
866     }
867     m_CaptureWidget = (CPDFSDK_Widget*)pFXAnnot;
868     m_bOnWidget = TRUE;
869     if (!m_bEnterWidget) {
870       m_bEnterWidget = TRUE;
871       m_bExitWidget = FALSE;
872       pAnnotHandlerMgr->Annot_OnMouseEnter(this, pFXAnnot, nFlag);
873     }
874     pAnnotHandlerMgr->Annot_OnMouseMove(this, pFXAnnot, nFlag, point);
875     return TRUE;
876   }
877   if (m_bOnWidget) {
878     m_bOnWidget = FALSE;
879     m_bExitWidget = TRUE;
880     m_bEnterWidget = FALSE;
881     if (m_CaptureWidget) {
882       pAnnotHandlerMgr->Annot_OnMouseExit(this, m_CaptureWidget, nFlag);
883       m_CaptureWidget = NULL;
884     }
885   }
886   return FALSE;
887 }
888
889 FX_BOOL CPDFSDK_PageView::OnMouseWheel(double deltaX,
890                                        double deltaY,
891                                        const CPDF_Point& point,
892                                        int nFlag) {
893   if (CPDFSDK_Annot* pAnnot = GetFXWidgetAtPoint(point.x, point.y)) {
894     CPDFDoc_Environment* pEnv = m_pSDKDoc->GetEnv();
895     CPDFSDK_AnnotHandlerMgr* pAnnotHandlerMgr = pEnv->GetAnnotHandlerMgr();
896     ASSERT(pAnnotHandlerMgr);
897     return pAnnotHandlerMgr->Annot_OnMouseWheel(this, pAnnot, nFlag,
898                                                 (int)deltaY, point);
899   }
900   return FALSE;
901 }
902
903 FX_BOOL CPDFSDK_PageView::OnChar(int nChar, FX_UINT nFlag) {
904   if (CPDFSDK_Annot* pAnnot = GetFocusAnnot()) {
905     CPDFDoc_Environment* pEnv = m_pSDKDoc->GetEnv();
906     CPDFSDK_AnnotHandlerMgr* pAnnotHandlerMgr = pEnv->GetAnnotHandlerMgr();
907     ASSERT(pAnnotHandlerMgr);
908     return pAnnotHandlerMgr->Annot_OnChar(pAnnot, nChar, nFlag);
909   }
910
911   return FALSE;
912 }
913
914 FX_BOOL CPDFSDK_PageView::OnKeyDown(int nKeyCode, int nFlag) {
915   if (CPDFSDK_Annot* pAnnot = GetFocusAnnot()) {
916     CPDFDoc_Environment* pEnv = m_pSDKDoc->GetEnv();
917     CPDFSDK_AnnotHandlerMgr* pAnnotHandlerMgr = pEnv->GetAnnotHandlerMgr();
918     ASSERT(pAnnotHandlerMgr);
919     return pAnnotHandlerMgr->Annot_OnKeyDown(pAnnot, nKeyCode, nFlag);
920   }
921   return FALSE;
922 }
923
924 FX_BOOL CPDFSDK_PageView::OnKeyUp(int nKeyCode, int nFlag) {
925   return FALSE;
926 }
927
928 void CPDFSDK_PageView::LoadFXAnnots() {
929   CPDFDoc_Environment* pEnv = m_pSDKDoc->GetEnv();
930
931   FX_BOOL enableAPUpdate = CPDF_InterForm::UpdatingAPEnabled();
932   // Disable the default AP construction.
933   CPDF_InterForm::EnableUpdateAP(FALSE);
934   m_pAnnotList = new CPDF_AnnotList(m_page);
935   CPDF_InterForm::EnableUpdateAP(enableAPUpdate);
936   int nCount = m_pAnnotList->Count();
937   SetLock(TRUE);
938   for (int i = 0; i < nCount; i++) {
939     CPDF_Annot* pPDFAnnot = m_pAnnotList->GetAt(i);
940     CPDF_Document* pDoc = GetPDFDocument();
941
942     CheckUnSupportAnnot(pDoc, pPDFAnnot);
943
944     CPDFSDK_AnnotHandlerMgr* pAnnotHandlerMgr = pEnv->GetAnnotHandlerMgr();
945     ASSERT(pAnnotHandlerMgr != NULL);
946
947     if (pAnnotHandlerMgr) {
948       CPDFSDK_Annot* pAnnot = pAnnotHandlerMgr->NewAnnot(pPDFAnnot, this);
949       if (!pAnnot)
950         continue;
951       m_fxAnnotArray.Add(pAnnot);
952
953       pAnnotHandlerMgr->Annot_OnLoad(pAnnot);
954     }
955   }
956   SetLock(FALSE);
957 }
958
959 void CPDFSDK_PageView::UpdateRects(CFX_RectArray& rects) {
960   for (int i = 0; i < rects.GetSize(); i++) {
961     CPDF_Rect rc = rects.GetAt(i);
962     CPDFDoc_Environment* pEnv = m_pSDKDoc->GetEnv();
963     pEnv->FFI_Invalidate(m_page, rc.left, rc.top, rc.right, rc.bottom);
964   }
965 }
966
967 void CPDFSDK_PageView::UpdateView(CPDFSDK_Annot* pAnnot) {
968   CPDF_Rect rcWindow = pAnnot->GetRect();
969   CPDFDoc_Environment* pEnv = m_pSDKDoc->GetEnv();
970   pEnv->FFI_Invalidate(m_page, rcWindow.left, rcWindow.top, rcWindow.right,
971                        rcWindow.bottom);
972 }
973
974 int CPDFSDK_PageView::GetPageIndex() {
975   if (m_page) {
976     CPDF_Dictionary* pDic = m_page->m_pFormDict;
977     CPDF_Document* pDoc = m_pSDKDoc->GetDocument();
978     if (pDoc && pDic) {
979       return pDoc->GetPageIndex(pDic->GetObjNum());
980     }
981   }
982   return -1;
983 }
984
985 FX_BOOL CPDFSDK_PageView::IsValidAnnot(void* p) {
986   if (p == NULL)
987     return FALSE;
988   int iCount = m_pAnnotList->Count();
989   for (int i = 0; i < iCount; i++) {
990     if (m_pAnnotList->GetAt(i) == p)
991       return TRUE;
992   }
993   return FALSE;
994 }
995
996 CPDFSDK_Annot* CPDFSDK_PageView::GetFocusAnnot() {
997   CPDFSDK_Annot* pFocusAnnot = m_pSDKDoc->GetFocusAnnot();
998   if (!pFocusAnnot)
999     return NULL;
1000
1001   for (int i = 0; i < m_fxAnnotArray.GetSize(); i++) {
1002     CPDFSDK_Annot* pAnnot = (CPDFSDK_Annot*)m_fxAnnotArray.GetAt(i);
1003     if (pAnnot == pFocusAnnot)
1004       return pAnnot;
1005   }
1006   return NULL;
1007 }