Clean: Fix some unneeded semi-colons and bad spacing.
[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(this);
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   CPDFDoc_Environment* pEnv = m_pSDKDoc->GetEnv();
638   int nAnnotCount = m_fxAnnotArray.GetSize();
639
640   for (int i = 0; i < nAnnotCount; i++) {
641     CPDFSDK_Annot* pAnnot = (CPDFSDK_Annot*)m_fxAnnotArray.GetAt(i);
642     // if there is a focused annot on the page, we should kill the focus first.
643     if (pAnnot == m_pSDKDoc->GetFocusAnnot())
644       KillFocusAnnot();
645     CPDFSDK_AnnotHandlerMgr* pAnnotHandlerMgr = pEnv->GetAnnotHandlerMgr();
646     ASSERT(pAnnotHandlerMgr);
647     pAnnotHandlerMgr->ReleaseAnnot(pAnnot);
648   }
649   m_fxAnnotArray.RemoveAll();
650
651   delete m_pAnnotList;
652   m_pAnnotList = NULL;
653
654   m_page->RemovePrivateData((void*)m_page);
655   if (m_bTakeOverPage) {
656     delete m_page;
657   }
658 }
659
660 void CPDFSDK_PageView::PageView_OnDraw(CFX_RenderDevice* pDevice,
661                                        CPDF_Matrix* pUser2Device,
662                                        CPDF_RenderOptions* pOptions) {
663   m_curMatrix = *pUser2Device;
664
665   CPDFDoc_Environment* pEnv = m_pSDKDoc->GetEnv();
666   CPDFSDK_AnnotIterator annotIterator(this, TRUE);
667   CPDFSDK_Annot* pSDKAnnot = nullptr;
668   int index = -1;
669   while ((pSDKAnnot = annotIterator.Next(index))) {
670     CPDFSDK_AnnotHandlerMgr* pAnnotHandlerMgr = pEnv->GetAnnotHandlerMgr();
671     ASSERT(pAnnotHandlerMgr);
672     pAnnotHandlerMgr->Annot_OnDraw(this, pSDKAnnot, pDevice, pUser2Device, 0);
673   }
674 }
675
676 CPDF_Annot* CPDFSDK_PageView::GetPDFAnnotAtPoint(FX_FLOAT pageX,
677                                                  FX_FLOAT pageY) {
678   int nCount = m_pAnnotList->Count();
679   for (int i = 0; i < nCount; i++) {
680     CPDF_Annot* pAnnot = m_pAnnotList->GetAt(i);
681     CFX_FloatRect annotRect;
682     pAnnot->GetRect(annotRect);
683     if (annotRect.Contains(pageX, pageY))
684       return pAnnot;
685   }
686   return NULL;
687 }
688
689 CPDF_Annot* CPDFSDK_PageView::GetPDFWidgetAtPoint(FX_FLOAT pageX,
690                                                   FX_FLOAT pageY) {
691   int nCount = m_pAnnotList->Count();
692   for (int i = 0; i < nCount; i++) {
693     CPDF_Annot* pAnnot = m_pAnnotList->GetAt(i);
694     if (pAnnot->GetSubType() == "Widget") {
695       CFX_FloatRect annotRect;
696       pAnnot->GetRect(annotRect);
697       if (annotRect.Contains(pageX, pageY))
698         return pAnnot;
699     }
700   }
701   return NULL;
702 }
703
704 CPDFSDK_Annot* CPDFSDK_PageView::GetFXAnnotAtPoint(FX_FLOAT pageX,
705                                                    FX_FLOAT pageY) {
706   CPDFSDK_AnnotIterator annotIterator(this, FALSE);
707   CPDFDoc_Environment* pEnv = m_pSDKDoc->GetEnv();
708   CPDFSDK_AnnotHandlerMgr* pAnnotMgr = pEnv->GetAnnotHandlerMgr();
709   CPDFSDK_Annot* pSDKAnnot = NULL;
710   int index = -1;
711   while ((pSDKAnnot = annotIterator.Next(index))) {
712     CPDF_Rect rc = pAnnotMgr->Annot_OnGetViewBBox(this, pSDKAnnot);
713     if (rc.Contains(pageX, pageY))
714       return pSDKAnnot;
715   }
716
717   return NULL;
718 }
719
720 CPDFSDK_Annot* CPDFSDK_PageView::GetFXWidgetAtPoint(FX_FLOAT pageX,
721                                                     FX_FLOAT pageY) {
722   CPDFSDK_AnnotIterator annotIterator(this, FALSE);
723   CPDFDoc_Environment* pEnv = m_pSDKDoc->GetEnv();
724   CPDFSDK_AnnotHandlerMgr* pAnnotMgr = pEnv->GetAnnotHandlerMgr();
725   CPDFSDK_Annot* pSDKAnnot = NULL;
726   int index = -1;
727   while ((pSDKAnnot = annotIterator.Next(index))) {
728     if (pSDKAnnot->GetType() == "Widget") {
729       pAnnotMgr->Annot_OnGetViewBBox(this, pSDKAnnot);
730       CPDF_Point point(pageX, pageY);
731       if (pAnnotMgr->Annot_OnHitTest(this, pSDKAnnot, point))
732         return pSDKAnnot;
733     }
734   }
735
736   return NULL;
737 }
738
739 FX_BOOL CPDFSDK_PageView::Annot_HasAppearance(CPDF_Annot* pAnnot) {
740   CPDF_Dictionary* pAnnotDic = pAnnot->GetAnnotDict();
741   if (pAnnotDic)
742     return pAnnotDic->KeyExist("AS");
743   return FALSE;
744 }
745
746 CPDFSDK_Annot* CPDFSDK_PageView::AddAnnot(CPDF_Annot* pPDFAnnot) {
747   CPDFDoc_Environment* pEnv = m_pSDKDoc->GetEnv();
748   ASSERT(pEnv);
749   CPDFSDK_AnnotHandlerMgr* pAnnotHandler = pEnv->GetAnnotHandlerMgr();
750
751   CPDFSDK_Annot* pSDKAnnot = NULL;
752
753   if (pAnnotHandler) {
754     pSDKAnnot = pAnnotHandler->NewAnnot(pPDFAnnot, this);
755   }
756   if (!pSDKAnnot)
757     return NULL;
758
759   m_fxAnnotArray.Add(pSDKAnnot);
760
761   if (pAnnotHandler) {
762     pAnnotHandler->Annot_OnCreate(pSDKAnnot);
763   }
764
765   return pSDKAnnot;
766 }
767
768 CPDFSDK_Annot* CPDFSDK_PageView::AddAnnot(CPDF_Dictionary* pDict) {
769   return pDict ? AddAnnot(pDict->GetString("Subtype"), pDict) : nullptr;
770 }
771
772 CPDFSDK_Annot* CPDFSDK_PageView::AddAnnot(const FX_CHAR* lpSubType,
773                                           CPDF_Dictionary* pDict) {
774   return NULL;
775 }
776
777 FX_BOOL CPDFSDK_PageView::DeleteAnnot(CPDFSDK_Annot* pAnnot) {
778   return FALSE;
779 }
780
781 CPDF_Document* CPDFSDK_PageView::GetPDFDocument() {
782   if (m_page) {
783     return m_page->m_pDocument;
784   }
785   return NULL;
786 }
787
788 int CPDFSDK_PageView::CountAnnots() {
789   return m_pAnnotList->Count();
790 }
791
792 CPDFSDK_Annot* CPDFSDK_PageView::GetAnnot(int nIndex) {
793   int nCount = m_fxAnnotArray.GetSize();
794   if (nIndex < 0 || nIndex >= nCount) {
795     return NULL;
796   }
797
798   return (CPDFSDK_Annot*)m_fxAnnotArray.GetAt(nIndex);
799 }
800
801 CPDFSDK_Annot* CPDFSDK_PageView::GetAnnotByDict(CPDF_Dictionary* pDict) {
802   int nCount = m_fxAnnotArray.GetSize();
803   for (int i = 0; i < nCount; i++) {
804     CPDFSDK_Annot* pAnnot = (CPDFSDK_Annot*)m_fxAnnotArray.GetAt(i);
805     if (pDict == pAnnot->GetPDFAnnot()->GetAnnotDict())
806       return pAnnot;
807   }
808   return NULL;
809 }
810
811 FX_BOOL CPDFSDK_PageView::OnLButtonDown(const CPDF_Point& point,
812                                         FX_UINT nFlag) {
813   CPDFDoc_Environment* pEnv = m_pSDKDoc->GetEnv();
814   ASSERT(pEnv);
815   CPDFSDK_Annot* pFXAnnot = GetFXWidgetAtPoint(point.x, point.y);
816   if (!pFXAnnot) {
817     KillFocusAnnot(nFlag);
818   } else {
819     CPDFSDK_AnnotHandlerMgr* pAnnotHandlerMgr = pEnv->GetAnnotHandlerMgr();
820     ASSERT(pAnnotHandlerMgr);
821
822     FX_BOOL bRet =
823         pAnnotHandlerMgr->Annot_OnLButtonDown(this, pFXAnnot, nFlag, point);
824     if (bRet) {
825       SetFocusAnnot(pFXAnnot);
826     }
827     return bRet;
828   }
829   return FALSE;
830 }
831
832 FX_BOOL CPDFSDK_PageView::OnLButtonUp(const CPDF_Point& point, FX_UINT nFlag) {
833   CPDFDoc_Environment* pEnv = m_pSDKDoc->GetEnv();
834   ASSERT(pEnv);
835   CPDFSDK_AnnotHandlerMgr* pAnnotHandlerMgr = pEnv->GetAnnotHandlerMgr();
836   ASSERT(pAnnotHandlerMgr);
837   CPDFSDK_Annot* pFXAnnot = GetFXWidgetAtPoint(point.x, point.y);
838   CPDFSDK_Annot* pFocusAnnot = GetFocusAnnot();
839   FX_BOOL bRet = FALSE;
840   if (pFocusAnnot && pFocusAnnot != pFXAnnot) {
841     // Last focus Annot gets a chance to handle the event.
842     bRet = pAnnotHandlerMgr->Annot_OnLButtonUp(this, pFocusAnnot, nFlag, point);
843   }
844   if (pFXAnnot && !bRet) {
845     bRet = pAnnotHandlerMgr->Annot_OnLButtonUp(this, pFXAnnot, nFlag, point);
846     return bRet;
847   }
848   return bRet;
849 }
850
851 FX_BOOL CPDFSDK_PageView::OnMouseMove(const CPDF_Point& point, int nFlag) {
852   CPDFDoc_Environment* pEnv = m_pSDKDoc->GetEnv();
853   CPDFSDK_AnnotHandlerMgr* pAnnotHandlerMgr = pEnv->GetAnnotHandlerMgr();
854   ASSERT(pAnnotHandlerMgr);
855   if (CPDFSDK_Annot* pFXAnnot = GetFXWidgetAtPoint(point.x, point.y)) {
856     if (m_CaptureWidget && m_CaptureWidget != pFXAnnot) {
857       m_bExitWidget = TRUE;
858       m_bEnterWidget = FALSE;
859       pAnnotHandlerMgr->Annot_OnMouseExit(this, m_CaptureWidget, nFlag);
860     }
861     m_CaptureWidget = (CPDFSDK_Widget*)pFXAnnot;
862     m_bOnWidget = TRUE;
863     if (!m_bEnterWidget) {
864       m_bEnterWidget = TRUE;
865       m_bExitWidget = FALSE;
866       pAnnotHandlerMgr->Annot_OnMouseEnter(this, pFXAnnot, nFlag);
867     }
868     pAnnotHandlerMgr->Annot_OnMouseMove(this, pFXAnnot, nFlag, point);
869     return TRUE;
870   }
871   if (m_bOnWidget) {
872     m_bOnWidget = FALSE;
873     m_bExitWidget = TRUE;
874     m_bEnterWidget = FALSE;
875     if (m_CaptureWidget) {
876       pAnnotHandlerMgr->Annot_OnMouseExit(this, m_CaptureWidget, nFlag);
877       m_CaptureWidget = NULL;
878     }
879   }
880   return FALSE;
881 }
882
883 FX_BOOL CPDFSDK_PageView::OnMouseWheel(double deltaX,
884                                        double deltaY,
885                                        const CPDF_Point& point,
886                                        int nFlag) {
887   if (CPDFSDK_Annot* pAnnot = GetFXWidgetAtPoint(point.x, point.y)) {
888     CPDFDoc_Environment* pEnv = m_pSDKDoc->GetEnv();
889     CPDFSDK_AnnotHandlerMgr* pAnnotHandlerMgr = pEnv->GetAnnotHandlerMgr();
890     ASSERT(pAnnotHandlerMgr);
891     return pAnnotHandlerMgr->Annot_OnMouseWheel(this, pAnnot, nFlag,
892                                                 (int)deltaY, point);
893   }
894   return FALSE;
895 }
896
897 FX_BOOL CPDFSDK_PageView::OnChar(int nChar, FX_UINT nFlag) {
898   if (CPDFSDK_Annot* pAnnot = GetFocusAnnot()) {
899     CPDFDoc_Environment* pEnv = m_pSDKDoc->GetEnv();
900     CPDFSDK_AnnotHandlerMgr* pAnnotHandlerMgr = pEnv->GetAnnotHandlerMgr();
901     ASSERT(pAnnotHandlerMgr);
902     return pAnnotHandlerMgr->Annot_OnChar(pAnnot, nChar, nFlag);
903   }
904
905   return FALSE;
906 }
907
908 FX_BOOL CPDFSDK_PageView::OnKeyDown(int nKeyCode, int nFlag) {
909   if (CPDFSDK_Annot* pAnnot = GetFocusAnnot()) {
910     CPDFDoc_Environment* pEnv = m_pSDKDoc->GetEnv();
911     CPDFSDK_AnnotHandlerMgr* pAnnotHandlerMgr = pEnv->GetAnnotHandlerMgr();
912     ASSERT(pAnnotHandlerMgr);
913     return pAnnotHandlerMgr->Annot_OnKeyDown(pAnnot, nKeyCode, nFlag);
914   }
915   return FALSE;
916 }
917
918 FX_BOOL CPDFSDK_PageView::OnKeyUp(int nKeyCode, int nFlag) {
919   //  if(CPDFSDK_Annot* pAnnot = GetFocusAnnot())
920   //  {
921   //      CFFL_IFormFiller* pIFormFiller = g_pFormFillApp->GetIFormFiller();
922   //      return pIFormFiller->OnKeyUp(pAnnot, nKeyCode, nFlag);
923   //  }
924   return FALSE;
925 }
926
927 extern void CheckUnSupportAnnot(CPDF_Document* pDoc, CPDF_Annot* pPDFAnnot);
928
929 void CPDFSDK_PageView::LoadFXAnnots() {
930   CPDFDoc_Environment* pEnv = m_pSDKDoc->GetEnv();
931
932   FX_BOOL enableAPUpdate = CPDF_InterForm::UpdatingAPEnabled();
933   // Disable the default AP construction.
934   CPDF_InterForm::EnableUpdateAP(FALSE);
935   m_pAnnotList = new CPDF_AnnotList(m_page);
936   CPDF_InterForm::EnableUpdateAP(enableAPUpdate);
937   int nCount = m_pAnnotList->Count();
938   SetLock(TRUE);
939   for (int i = 0; i < nCount; i++) {
940     CPDF_Annot* pPDFAnnot = m_pAnnotList->GetAt(i);
941     CPDF_Document* pDoc = GetPDFDocument();
942
943     CheckUnSupportAnnot(pDoc, pPDFAnnot);
944
945     CPDFSDK_AnnotHandlerMgr* pAnnotHandlerMgr = pEnv->GetAnnotHandlerMgr();
946     ASSERT(pAnnotHandlerMgr != NULL);
947
948     if (pAnnotHandlerMgr) {
949       CPDFSDK_Annot* pAnnot = pAnnotHandlerMgr->NewAnnot(pPDFAnnot, this);
950       if (!pAnnot)
951         continue;
952       m_fxAnnotArray.Add(pAnnot);
953
954       pAnnotHandlerMgr->Annot_OnLoad(pAnnot);
955     }
956   }
957   SetLock(FALSE);
958 }
959
960 void CPDFSDK_PageView::UpdateRects(CFX_RectArray& rects) {
961   for (int i = 0; i < rects.GetSize(); i++) {
962     CPDF_Rect rc = rects.GetAt(i);
963     CPDFDoc_Environment* pEnv = m_pSDKDoc->GetEnv();
964     pEnv->FFI_Invalidate(m_page, rc.left, rc.top, rc.right, rc.bottom);
965   }
966 }
967
968 void CPDFSDK_PageView::UpdateView(CPDFSDK_Annot* pAnnot) {
969   CPDF_Rect rcWindow = pAnnot->GetRect();
970   CPDFDoc_Environment* pEnv = m_pSDKDoc->GetEnv();
971   pEnv->FFI_Invalidate(m_page, rcWindow.left, rcWindow.top, rcWindow.right,
972                        rcWindow.bottom);
973 }
974
975 int CPDFSDK_PageView::GetPageIndex() {
976   if (m_page) {
977     CPDF_Dictionary* pDic = m_page->m_pFormDict;
978     CPDF_Document* pDoc = m_pSDKDoc->GetDocument();
979     if (pDoc && pDic) {
980       return pDoc->GetPageIndex(pDic->GetObjNum());
981     }
982   }
983   return -1;
984 }
985
986 FX_BOOL CPDFSDK_PageView::IsValidAnnot(void* p) {
987   if (p == NULL)
988     return FALSE;
989   int iCount = m_pAnnotList->Count();
990   for (int i = 0; i < iCount; i++) {
991     if (m_pAnnotList->GetAt(i) == p)
992       return TRUE;
993   }
994   return FALSE;
995 }
996
997 CPDFSDK_Annot* CPDFSDK_PageView::GetFocusAnnot() {
998   CPDFSDK_Annot* pFocusAnnot = m_pSDKDoc->GetFocusAnnot();
999   if (!pFocusAnnot)
1000     return NULL;
1001
1002   for (int i = 0; i < m_fxAnnotArray.GetSize(); i++) {
1003     CPDFSDK_Annot* pAnnot = (CPDFSDK_Annot*)m_fxAnnotArray.GetAt(i);
1004     if (pAnnot == pFocusAnnot)
1005       return pAnnot;
1006   }
1007   return NULL;
1008 }