Add type cast definitions for CPDF_Boolean.
[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/formfiller/FFL_FormFiller.h"
10 #include "../include/fsdk_define.h"
11 #include "../include/fsdk_mgr.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 CPDFDoc_Environment::CPDFDoc_Environment(CPDF_Document* pDoc,
207                                          FPDF_FORMFILLINFO* pFFinfo)
208     : m_pAnnotHandlerMgr(NULL),
209       m_pActionHandler(NULL),
210       m_pInfo(pFFinfo),
211       m_pSDKDoc(NULL),
212       m_pPDFDoc(pDoc),
213       m_pIFormFiller(NULL) {
214   m_pSysHandler = new CFX_SystemHandler(this);
215 }
216
217 CPDFDoc_Environment::~CPDFDoc_Environment() {
218   delete m_pIFormFiller;
219   m_pIFormFiller = NULL;
220
221   delete m_pSysHandler;
222   m_pSysHandler = NULL;
223
224   delete m_pAnnotHandlerMgr;
225   m_pAnnotHandlerMgr = NULL;
226
227   delete m_pActionHandler;
228   m_pActionHandler = NULL;
229 }
230
231 int CPDFDoc_Environment::JS_appAlert(const FX_WCHAR* Msg,
232                                      const FX_WCHAR* Title,
233                                      FX_UINT Type,
234                                      FX_UINT Icon) {
235   if (m_pInfo && m_pInfo->m_pJsPlatform && m_pInfo->m_pJsPlatform->app_alert) {
236     CFX_ByteString bsMsg = CFX_WideString(Msg).UTF16LE_Encode();
237     CFX_ByteString bsTitle = CFX_WideString(Title).UTF16LE_Encode();
238     FPDF_WIDESTRING pMsg = (FPDF_WIDESTRING)bsMsg.GetBuffer(bsMsg.GetLength());
239     FPDF_WIDESTRING pTitle =
240         (FPDF_WIDESTRING)bsTitle.GetBuffer(bsTitle.GetLength());
241     int ret = m_pInfo->m_pJsPlatform->app_alert(m_pInfo->m_pJsPlatform, pMsg,
242                                                 pTitle, Type, Icon);
243     bsMsg.ReleaseBuffer();
244     bsTitle.ReleaseBuffer();
245     return ret;
246   }
247   return -1;
248 }
249
250 int CPDFDoc_Environment::JS_appResponse(const FX_WCHAR* Question,
251                                         const FX_WCHAR* Title,
252                                         const FX_WCHAR* Default,
253                                         const FX_WCHAR* cLabel,
254                                         FPDF_BOOL bPassword,
255                                         void* response,
256                                         int length) {
257   if (m_pInfo && m_pInfo->m_pJsPlatform &&
258       m_pInfo->m_pJsPlatform->app_response) {
259     CFX_ByteString bsQuestion = CFX_WideString(Question).UTF16LE_Encode();
260     CFX_ByteString bsTitle = CFX_WideString(Title).UTF16LE_Encode();
261     CFX_ByteString bsDefault = CFX_WideString(Default).UTF16LE_Encode();
262     CFX_ByteString bsLabel = CFX_WideString(cLabel).UTF16LE_Encode();
263     FPDF_WIDESTRING pQuestion =
264         (FPDF_WIDESTRING)bsQuestion.GetBuffer(bsQuestion.GetLength());
265     FPDF_WIDESTRING pTitle =
266         (FPDF_WIDESTRING)bsTitle.GetBuffer(bsTitle.GetLength());
267     FPDF_WIDESTRING pDefault =
268         (FPDF_WIDESTRING)bsDefault.GetBuffer(bsDefault.GetLength());
269     FPDF_WIDESTRING pLabel =
270         (FPDF_WIDESTRING)bsLabel.GetBuffer(bsLabel.GetLength());
271     int ret = m_pInfo->m_pJsPlatform->app_response(
272         m_pInfo->m_pJsPlatform, pQuestion, pTitle, pDefault, pLabel, bPassword,
273         response, length);
274     bsQuestion.ReleaseBuffer();
275     bsTitle.ReleaseBuffer();
276     bsDefault.ReleaseBuffer();
277     bsLabel.ReleaseBuffer();
278     return ret;
279   }
280   return -1;
281 }
282
283 CFX_WideString CPDFDoc_Environment::JS_fieldBrowse() {
284   if (!m_pInfo || !m_pInfo->m_pJsPlatform ||
285       !m_pInfo->m_pJsPlatform->Field_browse) {
286     return L"";
287   }
288
289   const int nRequiredLen =
290       m_pInfo->m_pJsPlatform->Field_browse(m_pInfo->m_pJsPlatform, nullptr, 0);
291   if (nRequiredLen <= 0)
292     return L"";
293
294   nonstd::unique_ptr<char[]> pBuff(new char[nRequiredLen]);
295   memset(pBuff.get(), 0, nRequiredLen);
296   const int nActualLen = m_pInfo->m_pJsPlatform->Field_browse(
297       m_pInfo->m_pJsPlatform, pBuff.get(), nRequiredLen);
298   if (nActualLen <= 0 || nActualLen > nRequiredLen)
299     return L"";
300
301   CFX_ByteString bsRet = CFX_ByteString(pBuff.get(), nActualLen);
302   CFX_WideString wsRet = CFX_WideString::FromLocal(bsRet);
303   return wsRet;
304 }
305
306 CFX_WideString CPDFDoc_Environment::JS_docGetFilePath() {
307   if (!m_pInfo || !m_pInfo->m_pJsPlatform ||
308       !m_pInfo->m_pJsPlatform->Doc_getFilePath) {
309     return L"";
310   }
311
312   const int nRequiredLen = m_pInfo->m_pJsPlatform->Doc_getFilePath(
313       m_pInfo->m_pJsPlatform, nullptr, 0);
314   if (nRequiredLen <= 0)
315     return L"";
316
317   nonstd::unique_ptr<char[]> pBuff(new char[nRequiredLen]);
318   memset(pBuff.get(), 0, nRequiredLen);
319   const int nActualLen = m_pInfo->m_pJsPlatform->Doc_getFilePath(
320       m_pInfo->m_pJsPlatform, pBuff.get(), nRequiredLen);
321   if (nActualLen <= 0 || nActualLen > nRequiredLen)
322     return L"";
323
324   CFX_ByteString bsRet = CFX_ByteString(pBuff.get(), nActualLen);
325   CFX_WideString wsRet = CFX_WideString::FromLocal(bsRet);
326   return wsRet;
327 }
328
329 void CPDFDoc_Environment::JS_docSubmitForm(void* formData,
330                                            int length,
331                                            const FX_WCHAR* URL) {
332   if (m_pInfo && m_pInfo->m_pJsPlatform &&
333       m_pInfo->m_pJsPlatform->Doc_submitForm) {
334     CFX_ByteString bsDestination = CFX_WideString(URL).UTF16LE_Encode();
335     FPDF_WIDESTRING pDestination =
336         (FPDF_WIDESTRING)bsDestination.GetBuffer(bsDestination.GetLength());
337     m_pInfo->m_pJsPlatform->Doc_submitForm(m_pInfo->m_pJsPlatform, formData,
338                                            length, pDestination);
339     bsDestination.ReleaseBuffer();
340   }
341 }
342
343 void CPDFDoc_Environment::JS_docmailForm(void* mailData,
344                                          int length,
345                                          FPDF_BOOL bUI,
346                                          const FX_WCHAR* To,
347                                          const FX_WCHAR* Subject,
348                                          const FX_WCHAR* CC,
349                                          const FX_WCHAR* BCC,
350                                          const FX_WCHAR* Msg) {
351   if (m_pInfo && m_pInfo->m_pJsPlatform && m_pInfo->m_pJsPlatform->Doc_mail) {
352     CFX_ByteString bsTo = CFX_WideString(To).UTF16LE_Encode();
353     CFX_ByteString bsCC = CFX_WideString(Subject).UTF16LE_Encode();
354     CFX_ByteString bsBcc = CFX_WideString(BCC).UTF16LE_Encode();
355     CFX_ByteString bsSubject = CFX_WideString(Subject).UTF16LE_Encode();
356     CFX_ByteString bsMsg = CFX_WideString(Msg).UTF16LE_Encode();
357     FPDF_WIDESTRING pTo = (FPDF_WIDESTRING)bsTo.GetBuffer(bsTo.GetLength());
358     FPDF_WIDESTRING pCC = (FPDF_WIDESTRING)bsCC.GetBuffer(bsCC.GetLength());
359     FPDF_WIDESTRING pBcc = (FPDF_WIDESTRING)bsBcc.GetBuffer(bsBcc.GetLength());
360     FPDF_WIDESTRING pSubject =
361         (FPDF_WIDESTRING)bsSubject.GetBuffer(bsSubject.GetLength());
362     FPDF_WIDESTRING pMsg = (FPDF_WIDESTRING)bsMsg.GetBuffer(bsMsg.GetLength());
363     m_pInfo->m_pJsPlatform->Doc_mail(m_pInfo->m_pJsPlatform, mailData, length,
364                                      bUI, pTo, pSubject, pCC, pBcc, pMsg);
365     bsTo.ReleaseBuffer();
366     bsCC.ReleaseBuffer();
367     bsBcc.ReleaseBuffer();
368     bsSubject.ReleaseBuffer();
369     bsMsg.ReleaseBuffer();
370   }
371 }
372
373 IJS_Runtime* CPDFDoc_Environment::GetJSRuntime() {
374   if (!IsJSInitiated())
375     return NULL;
376   if (!m_pJSRuntime)
377     m_pJSRuntime.reset(IJS_Runtime::Create(this));
378   return m_pJSRuntime.get();
379 }
380
381 CPDFSDK_AnnotHandlerMgr* CPDFDoc_Environment::GetAnnotHandlerMgr() {
382   if (!m_pAnnotHandlerMgr)
383     m_pAnnotHandlerMgr = new CPDFSDK_AnnotHandlerMgr(this);
384   return m_pAnnotHandlerMgr;
385 }
386
387 CPDFSDK_ActionHandler* CPDFDoc_Environment::GetActionHander() {
388   if (!m_pActionHandler)
389     m_pActionHandler = new CPDFSDK_ActionHandler();
390   return m_pActionHandler;
391 }
392
393 CFFL_IFormFiller* CPDFDoc_Environment::GetIFormFiller() {
394   if (!m_pIFormFiller)
395     m_pIFormFiller = new CFFL_IFormFiller(this);
396   return m_pIFormFiller;
397 }
398
399 CPDFSDK_Document::CPDFSDK_Document(CPDF_Document* pDoc,
400                                    CPDFDoc_Environment* pEnv)
401     : m_pDoc(pDoc),
402       m_pInterForm(nullptr),
403       m_pFocusAnnot(nullptr),
404       m_pEnv(pEnv),
405       m_pOccontent(nullptr),
406       m_bChangeMask(FALSE) {}
407
408 CPDFSDK_Document::~CPDFSDK_Document() {
409   for (auto& it : m_pageMap)
410     delete it.second;
411   m_pageMap.clear();
412
413   delete m_pInterForm;
414   m_pInterForm = nullptr;
415
416   delete m_pOccontent;
417   m_pOccontent = nullptr;
418 }
419
420 CPDFSDK_PageView* CPDFSDK_Document::GetPageView(CPDF_Page* pPDFPage,
421                                                 FX_BOOL ReNew) {
422   auto it = m_pageMap.find(pPDFPage);
423   if (it != m_pageMap.end())
424     return it->second;
425
426   if (!ReNew)
427     return nullptr;
428
429   CPDFSDK_PageView* pPageView = new CPDFSDK_PageView(this, pPDFPage);
430   m_pageMap[pPDFPage] = pPageView;
431   // Delay to load all the annotations, to avoid endless loop.
432   pPageView->LoadFXAnnots();
433   return pPageView;
434 }
435
436 CPDFSDK_PageView* CPDFSDK_Document::GetCurrentView() {
437   CPDF_Page* pPage = (CPDF_Page*)m_pEnv->FFI_GetCurrentPage(m_pDoc);
438   return pPage ? GetPageView(pPage, TRUE) : nullptr;
439 }
440
441 CPDFSDK_PageView* CPDFSDK_Document::GetPageView(int nIndex) {
442   CPDF_Page* pTempPage = (CPDF_Page*)m_pEnv->FFI_GetPage(m_pDoc, nIndex);
443   if (!pTempPage)
444     return nullptr;
445
446   auto it = m_pageMap.find(pTempPage);
447   return it->second;
448 }
449
450 void CPDFSDK_Document::ProcJavascriptFun() {
451   CPDF_Document* pPDFDoc = GetDocument();
452   CPDF_DocJSActions docJS(pPDFDoc);
453   int iCount = docJS.CountJSActions();
454   if (iCount < 1)
455     return;
456   for (int i = 0; i < iCount; i++) {
457     CFX_ByteString csJSName;
458     CPDF_Action jsAction = docJS.GetJSAction(i, csJSName);
459     if (m_pEnv->GetActionHander())
460       m_pEnv->GetActionHander()->DoAction_JavaScript(
461           jsAction, CFX_WideString::FromLocal(csJSName), this);
462   }
463 }
464
465 FX_BOOL CPDFSDK_Document::ProcOpenAction() {
466   if (!m_pDoc)
467     return FALSE;
468
469   CPDF_Dictionary* pRoot = m_pDoc->GetRoot();
470   if (!pRoot)
471     return FALSE;
472
473   CPDF_Object* pOpenAction = pRoot->GetDict("OpenAction");
474   if (!pOpenAction)
475     pOpenAction = pRoot->GetArray("OpenAction");
476
477   if (!pOpenAction)
478     return FALSE;
479
480   if (pOpenAction->GetType() == PDFOBJ_ARRAY)
481     return TRUE;
482
483   if (CPDF_Dictionary* pDict = pOpenAction->AsDictionary()) {
484     CPDF_Action action(pDict);
485     if (m_pEnv->GetActionHander())
486       m_pEnv->GetActionHander()->DoAction_DocOpen(action, this);
487     return TRUE;
488   }
489   return FALSE;
490 }
491
492 CPDF_OCContext* CPDFSDK_Document::GetOCContext() {
493   if (!m_pOccontent)
494     m_pOccontent = new CPDF_OCContext(m_pDoc);
495   return m_pOccontent;
496 }
497
498 void CPDFSDK_Document::ReMovePageView(CPDF_Page* pPDFPage) {
499   auto it = m_pageMap.find(pPDFPage);
500   if (it == m_pageMap.end())
501     return;
502
503   CPDFSDK_PageView* pPageView = it->second;
504   if (pPageView->IsLocked())
505     return;
506
507   delete pPageView;
508   m_pageMap.erase(it);
509 }
510
511 CPDF_Page* CPDFSDK_Document::GetPage(int nIndex) {
512   CPDF_Page* pTempPage = (CPDF_Page*)m_pEnv->FFI_GetPage(m_pDoc, nIndex);
513   if (!pTempPage)
514     return NULL;
515   return pTempPage;
516 }
517
518 CPDFSDK_InterForm* CPDFSDK_Document::GetInterForm() {
519   if (!m_pInterForm)
520     m_pInterForm = new CPDFSDK_InterForm(this);
521   return m_pInterForm;
522 }
523
524 void CPDFSDK_Document::UpdateAllViews(CPDFSDK_PageView* pSender,
525                                       CPDFSDK_Annot* pAnnot) {
526   for (const auto& it : m_pageMap) {
527     CPDFSDK_PageView* pPageView = it.second;
528     if (pPageView != pSender) {
529       pPageView->UpdateView(pAnnot);
530     }
531   }
532 }
533
534 CPDFSDK_Annot* CPDFSDK_Document::GetFocusAnnot() {
535   return m_pFocusAnnot;
536 }
537
538 FX_BOOL CPDFSDK_Document::SetFocusAnnot(CPDFSDK_Annot* pAnnot, FX_UINT nFlag) {
539   if (m_pFocusAnnot == pAnnot)
540     return TRUE;
541
542   if (m_pFocusAnnot) {
543     if (!KillFocusAnnot(nFlag))
544       return FALSE;
545   }
546   CPDFSDK_PageView* pPageView = pAnnot->GetPageView();
547   if (pAnnot && pPageView->IsValid()) {
548     CPDFSDK_AnnotHandlerMgr* pAnnotHandler = m_pEnv->GetAnnotHandlerMgr();
549
550     if (pAnnotHandler && !m_pFocusAnnot) {
551       if (!pAnnotHandler->Annot_OnSetFocus(pAnnot, nFlag))
552         return FALSE;
553       if (!m_pFocusAnnot) {
554         m_pFocusAnnot = pAnnot;
555         return TRUE;
556       }
557     }
558   }
559   return FALSE;
560 }
561
562 FX_BOOL CPDFSDK_Document::KillFocusAnnot(FX_UINT nFlag) {
563   if (m_pFocusAnnot) {
564     CPDFSDK_AnnotHandlerMgr* pAnnotHandler = m_pEnv->GetAnnotHandlerMgr();
565     if (pAnnotHandler) {
566       CPDFSDK_Annot* pFocusAnnot = m_pFocusAnnot;
567       m_pFocusAnnot = NULL;
568       if (pAnnotHandler->Annot_OnKillFocus(pFocusAnnot, nFlag)) {
569         if (pFocusAnnot->GetType() == FX_BSTRC("Widget")) {
570           CPDFSDK_Widget* pWidget = (CPDFSDK_Widget*)pFocusAnnot;
571           int nFieldType = pWidget->GetFieldType();
572           if (FIELDTYPE_TEXTFIELD == nFieldType ||
573               FIELDTYPE_COMBOBOX == nFieldType)
574             m_pEnv->FFI_OnSetFieldInputFocus(NULL, NULL, 0, FALSE);
575         }
576
577         if (!m_pFocusAnnot)
578           return TRUE;
579       } else {
580         m_pFocusAnnot = pFocusAnnot;
581       }
582     }
583   }
584   return FALSE;
585 }
586
587 void CPDFSDK_Document::OnCloseDocument() {
588   KillFocusAnnot();
589 }
590
591 FX_BOOL CPDFSDK_Document::GetPermissions(int nFlag) {
592   FX_DWORD dwPermissions = m_pDoc->GetUserPermissions();
593   return dwPermissions & nFlag;
594 }
595
596 IJS_Runtime* CPDFSDK_Document::GetJsRuntime() {
597   ASSERT(m_pEnv != NULL);
598   return m_pEnv->GetJSRuntime();
599 }
600
601 CFX_WideString CPDFSDK_Document::GetPath() {
602   ASSERT(m_pEnv != NULL);
603   return m_pEnv->JS_docGetFilePath();
604 }
605
606 CPDFSDK_PageView::CPDFSDK_PageView(CPDFSDK_Document* pSDKDoc, CPDF_Page* page)
607     : m_page(page), m_pSDKDoc(pSDKDoc) {
608   CPDFSDK_InterForm* pInterForm = pSDKDoc->GetInterForm();
609   if (pInterForm) {
610     CPDF_InterForm* pPDFInterForm = pInterForm->GetInterForm();
611     pPDFInterForm->FixPageFields(page);
612   }
613   m_page->SetPrivateData((void*)m_page, (void*)this, NULL);
614   m_fxAnnotArray.RemoveAll();
615
616   m_bEnterWidget = FALSE;
617   m_bExitWidget = FALSE;
618   m_bOnWidget = FALSE;
619   m_CaptureWidget = NULL;
620   m_bValid = FALSE;
621   m_bLocked = FALSE;
622   m_bTakeOverPage = FALSE;
623 }
624
625 CPDFSDK_PageView::~CPDFSDK_PageView() {
626   // if there is a focused annot on the page, we should kill the focus first.
627   if (CPDFSDK_Annot* focusedAnnot = m_pSDKDoc->GetFocusAnnot()) {
628     for (int i = 0, count = m_fxAnnotArray.GetSize(); i < count; i++) {
629       CPDFSDK_Annot* pAnnot = (CPDFSDK_Annot*)m_fxAnnotArray.GetAt(i);
630       if (pAnnot == focusedAnnot) {
631         KillFocusAnnot();
632         break;
633       }
634     }
635   }
636
637   CPDFDoc_Environment* pEnv = m_pSDKDoc->GetEnv();
638   CPDFSDK_AnnotHandlerMgr* pAnnotHandlerMgr = pEnv->GetAnnotHandlerMgr();
639   ASSERT(pAnnotHandlerMgr);
640   for (int i = 0, count = m_fxAnnotArray.GetSize(); i < count; i++) {
641     CPDFSDK_Annot* pAnnot = (CPDFSDK_Annot*)m_fxAnnotArray.GetAt(i);
642     pAnnotHandlerMgr->ReleaseAnnot(pAnnot);
643   }
644   m_fxAnnotArray.RemoveAll();
645
646   delete m_pAnnotList;
647   m_pAnnotList = NULL;
648
649   m_page->RemovePrivateData((void*)m_page);
650   if (m_bTakeOverPage) {
651     delete m_page;
652   }
653 }
654
655 void CPDFSDK_PageView::PageView_OnDraw(CFX_RenderDevice* pDevice,
656                                        CPDF_Matrix* pUser2Device,
657                                        CPDF_RenderOptions* pOptions) {
658   m_curMatrix = *pUser2Device;
659
660   CPDFDoc_Environment* pEnv = m_pSDKDoc->GetEnv();
661   CPDFSDK_AnnotIterator annotIterator(this, TRUE);
662   int index = -1;
663   while (CPDFSDK_Annot* pSDKAnnot = annotIterator.Next(index)) {
664     CPDFSDK_AnnotHandlerMgr* pAnnotHandlerMgr = pEnv->GetAnnotHandlerMgr();
665     pAnnotHandlerMgr->Annot_OnDraw(this, pSDKAnnot, pDevice, pUser2Device, 0);
666   }
667 }
668
669 CPDF_Annot* CPDFSDK_PageView::GetPDFAnnotAtPoint(FX_FLOAT pageX,
670                                                  FX_FLOAT pageY) {
671   int nCount = m_pAnnotList->Count();
672   for (int i = 0; i < nCount; i++) {
673     CPDF_Annot* pAnnot = m_pAnnotList->GetAt(i);
674     CFX_FloatRect annotRect;
675     pAnnot->GetRect(annotRect);
676     if (annotRect.Contains(pageX, pageY))
677       return pAnnot;
678   }
679   return NULL;
680 }
681
682 CPDF_Annot* CPDFSDK_PageView::GetPDFWidgetAtPoint(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     if (pAnnot->GetSubType() == "Widget") {
688       CFX_FloatRect annotRect;
689       pAnnot->GetRect(annotRect);
690       if (annotRect.Contains(pageX, pageY))
691         return pAnnot;
692     }
693   }
694   return NULL;
695 }
696
697 CPDFSDK_Annot* CPDFSDK_PageView::GetFXAnnotAtPoint(FX_FLOAT pageX,
698                                                    FX_FLOAT pageY) {
699   CPDFSDK_AnnotIterator annotIterator(this, FALSE);
700   CPDFDoc_Environment* pEnv = m_pSDKDoc->GetEnv();
701   CPDFSDK_AnnotHandlerMgr* pAnnotMgr = pEnv->GetAnnotHandlerMgr();
702   CPDFSDK_Annot* pSDKAnnot = NULL;
703   int index = -1;
704   while ((pSDKAnnot = annotIterator.Next(index))) {
705     CPDF_Rect rc = pAnnotMgr->Annot_OnGetViewBBox(this, pSDKAnnot);
706     if (rc.Contains(pageX, pageY))
707       return pSDKAnnot;
708   }
709
710   return NULL;
711 }
712
713 CPDFSDK_Annot* CPDFSDK_PageView::GetFXWidgetAtPoint(FX_FLOAT pageX,
714                                                     FX_FLOAT pageY) {
715   CPDFSDK_AnnotIterator annotIterator(this, FALSE);
716   CPDFDoc_Environment* pEnv = m_pSDKDoc->GetEnv();
717   CPDFSDK_AnnotHandlerMgr* pAnnotMgr = pEnv->GetAnnotHandlerMgr();
718   CPDFSDK_Annot* pSDKAnnot = NULL;
719   int index = -1;
720   while ((pSDKAnnot = annotIterator.Next(index))) {
721     if (pSDKAnnot->GetType() == "Widget") {
722       pAnnotMgr->Annot_OnGetViewBBox(this, pSDKAnnot);
723       CPDF_Point point(pageX, pageY);
724       if (pAnnotMgr->Annot_OnHitTest(this, pSDKAnnot, point))
725         return pSDKAnnot;
726     }
727   }
728
729   return NULL;
730 }
731
732 FX_BOOL CPDFSDK_PageView::Annot_HasAppearance(CPDF_Annot* pAnnot) {
733   CPDF_Dictionary* pAnnotDic = pAnnot->GetAnnotDict();
734   if (pAnnotDic)
735     return pAnnotDic->KeyExist("AS");
736   return FALSE;
737 }
738
739 CPDFSDK_Annot* CPDFSDK_PageView::AddAnnot(CPDF_Annot* pPDFAnnot) {
740   CPDFDoc_Environment* pEnv = m_pSDKDoc->GetEnv();
741   ASSERT(pEnv);
742   CPDFSDK_AnnotHandlerMgr* pAnnotHandler = pEnv->GetAnnotHandlerMgr();
743
744   CPDFSDK_Annot* pSDKAnnot = NULL;
745
746   if (pAnnotHandler) {
747     pSDKAnnot = pAnnotHandler->NewAnnot(pPDFAnnot, this);
748   }
749   if (!pSDKAnnot)
750     return NULL;
751
752   m_fxAnnotArray.Add(pSDKAnnot);
753
754   if (pAnnotHandler) {
755     pAnnotHandler->Annot_OnCreate(pSDKAnnot);
756   }
757
758   return pSDKAnnot;
759 }
760
761 CPDFSDK_Annot* CPDFSDK_PageView::AddAnnot(CPDF_Dictionary* pDict) {
762   return pDict ? AddAnnot(pDict->GetString("Subtype"), pDict) : nullptr;
763 }
764
765 CPDFSDK_Annot* CPDFSDK_PageView::AddAnnot(const FX_CHAR* lpSubType,
766                                           CPDF_Dictionary* pDict) {
767   return NULL;
768 }
769
770 FX_BOOL CPDFSDK_PageView::DeleteAnnot(CPDFSDK_Annot* pAnnot) {
771   return FALSE;
772 }
773
774 CPDF_Document* CPDFSDK_PageView::GetPDFDocument() {
775   if (m_page) {
776     return m_page->m_pDocument;
777   }
778   return NULL;
779 }
780
781 int CPDFSDK_PageView::CountAnnots() {
782   return m_pAnnotList->Count();
783 }
784
785 CPDFSDK_Annot* CPDFSDK_PageView::GetAnnot(int nIndex) {
786   int nCount = m_fxAnnotArray.GetSize();
787   if (nIndex < 0 || nIndex >= nCount) {
788     return NULL;
789   }
790
791   return (CPDFSDK_Annot*)m_fxAnnotArray.GetAt(nIndex);
792 }
793
794 CPDFSDK_Annot* CPDFSDK_PageView::GetAnnotByDict(CPDF_Dictionary* pDict) {
795   int nCount = m_fxAnnotArray.GetSize();
796   for (int i = 0; i < nCount; i++) {
797     CPDFSDK_Annot* pAnnot = (CPDFSDK_Annot*)m_fxAnnotArray.GetAt(i);
798     if (pDict == pAnnot->GetPDFAnnot()->GetAnnotDict())
799       return pAnnot;
800   }
801   return NULL;
802 }
803
804 FX_BOOL CPDFSDK_PageView::OnLButtonDown(const CPDF_Point& point,
805                                         FX_UINT nFlag) {
806   CPDFDoc_Environment* pEnv = m_pSDKDoc->GetEnv();
807   ASSERT(pEnv);
808   CPDFSDK_Annot* pFXAnnot = GetFXWidgetAtPoint(point.x, point.y);
809   if (!pFXAnnot) {
810     KillFocusAnnot(nFlag);
811   } else {
812     CPDFSDK_AnnotHandlerMgr* pAnnotHandlerMgr = pEnv->GetAnnotHandlerMgr();
813     ASSERT(pAnnotHandlerMgr);
814
815     FX_BOOL bRet =
816         pAnnotHandlerMgr->Annot_OnLButtonDown(this, pFXAnnot, nFlag, point);
817     if (bRet) {
818       SetFocusAnnot(pFXAnnot);
819     }
820     return bRet;
821   }
822   return FALSE;
823 }
824
825 FX_BOOL CPDFSDK_PageView::OnLButtonUp(const CPDF_Point& point, FX_UINT nFlag) {
826   CPDFDoc_Environment* pEnv = m_pSDKDoc->GetEnv();
827   ASSERT(pEnv);
828   CPDFSDK_AnnotHandlerMgr* pAnnotHandlerMgr = pEnv->GetAnnotHandlerMgr();
829   ASSERT(pAnnotHandlerMgr);
830   CPDFSDK_Annot* pFXAnnot = GetFXWidgetAtPoint(point.x, point.y);
831   CPDFSDK_Annot* pFocusAnnot = GetFocusAnnot();
832   FX_BOOL bRet = FALSE;
833   if (pFocusAnnot && pFocusAnnot != pFXAnnot) {
834     // Last focus Annot gets a chance to handle the event.
835     bRet = pAnnotHandlerMgr->Annot_OnLButtonUp(this, pFocusAnnot, nFlag, point);
836   }
837   if (pFXAnnot && !bRet) {
838     bRet = pAnnotHandlerMgr->Annot_OnLButtonUp(this, pFXAnnot, nFlag, point);
839     return bRet;
840   }
841   return bRet;
842 }
843
844 FX_BOOL CPDFSDK_PageView::OnMouseMove(const CPDF_Point& point, int nFlag) {
845   CPDFDoc_Environment* pEnv = m_pSDKDoc->GetEnv();
846   CPDFSDK_AnnotHandlerMgr* pAnnotHandlerMgr = pEnv->GetAnnotHandlerMgr();
847   ASSERT(pAnnotHandlerMgr);
848   if (CPDFSDK_Annot* pFXAnnot = GetFXWidgetAtPoint(point.x, point.y)) {
849     if (m_CaptureWidget && m_CaptureWidget != pFXAnnot) {
850       m_bExitWidget = TRUE;
851       m_bEnterWidget = FALSE;
852       pAnnotHandlerMgr->Annot_OnMouseExit(this, m_CaptureWidget, nFlag);
853     }
854     m_CaptureWidget = (CPDFSDK_Widget*)pFXAnnot;
855     m_bOnWidget = TRUE;
856     if (!m_bEnterWidget) {
857       m_bEnterWidget = TRUE;
858       m_bExitWidget = FALSE;
859       pAnnotHandlerMgr->Annot_OnMouseEnter(this, pFXAnnot, nFlag);
860     }
861     pAnnotHandlerMgr->Annot_OnMouseMove(this, pFXAnnot, nFlag, point);
862     return TRUE;
863   }
864   if (m_bOnWidget) {
865     m_bOnWidget = FALSE;
866     m_bExitWidget = TRUE;
867     m_bEnterWidget = FALSE;
868     if (m_CaptureWidget) {
869       pAnnotHandlerMgr->Annot_OnMouseExit(this, m_CaptureWidget, nFlag);
870       m_CaptureWidget = NULL;
871     }
872   }
873   return FALSE;
874 }
875
876 FX_BOOL CPDFSDK_PageView::OnMouseWheel(double deltaX,
877                                        double deltaY,
878                                        const CPDF_Point& point,
879                                        int nFlag) {
880   if (CPDFSDK_Annot* pAnnot = GetFXWidgetAtPoint(point.x, point.y)) {
881     CPDFDoc_Environment* pEnv = m_pSDKDoc->GetEnv();
882     CPDFSDK_AnnotHandlerMgr* pAnnotHandlerMgr = pEnv->GetAnnotHandlerMgr();
883     ASSERT(pAnnotHandlerMgr);
884     return pAnnotHandlerMgr->Annot_OnMouseWheel(this, pAnnot, nFlag,
885                                                 (int)deltaY, point);
886   }
887   return FALSE;
888 }
889
890 FX_BOOL CPDFSDK_PageView::OnChar(int nChar, FX_UINT nFlag) {
891   if (CPDFSDK_Annot* pAnnot = GetFocusAnnot()) {
892     CPDFDoc_Environment* pEnv = m_pSDKDoc->GetEnv();
893     CPDFSDK_AnnotHandlerMgr* pAnnotHandlerMgr = pEnv->GetAnnotHandlerMgr();
894     ASSERT(pAnnotHandlerMgr);
895     return pAnnotHandlerMgr->Annot_OnChar(pAnnot, nChar, nFlag);
896   }
897
898   return FALSE;
899 }
900
901 FX_BOOL CPDFSDK_PageView::OnKeyDown(int nKeyCode, int nFlag) {
902   if (CPDFSDK_Annot* pAnnot = GetFocusAnnot()) {
903     CPDFDoc_Environment* pEnv = m_pSDKDoc->GetEnv();
904     CPDFSDK_AnnotHandlerMgr* pAnnotHandlerMgr = pEnv->GetAnnotHandlerMgr();
905     ASSERT(pAnnotHandlerMgr);
906     return pAnnotHandlerMgr->Annot_OnKeyDown(pAnnot, nKeyCode, nFlag);
907   }
908   return FALSE;
909 }
910
911 FX_BOOL CPDFSDK_PageView::OnKeyUp(int nKeyCode, int nFlag) {
912   return FALSE;
913 }
914
915 void CPDFSDK_PageView::LoadFXAnnots() {
916   CPDFDoc_Environment* pEnv = m_pSDKDoc->GetEnv();
917
918   FX_BOOL enableAPUpdate = CPDF_InterForm::UpdatingAPEnabled();
919   // Disable the default AP construction.
920   CPDF_InterForm::EnableUpdateAP(FALSE);
921   m_pAnnotList = new CPDF_AnnotList(m_page);
922   CPDF_InterForm::EnableUpdateAP(enableAPUpdate);
923   int nCount = m_pAnnotList->Count();
924   SetLock(TRUE);
925   for (int i = 0; i < nCount; i++) {
926     CPDF_Annot* pPDFAnnot = m_pAnnotList->GetAt(i);
927     CPDF_Document* pDoc = GetPDFDocument();
928
929     CheckUnSupportAnnot(pDoc, pPDFAnnot);
930
931     CPDFSDK_AnnotHandlerMgr* pAnnotHandlerMgr = pEnv->GetAnnotHandlerMgr();
932     ASSERT(pAnnotHandlerMgr != NULL);
933
934     if (pAnnotHandlerMgr) {
935       CPDFSDK_Annot* pAnnot = pAnnotHandlerMgr->NewAnnot(pPDFAnnot, this);
936       if (!pAnnot)
937         continue;
938       m_fxAnnotArray.Add(pAnnot);
939
940       pAnnotHandlerMgr->Annot_OnLoad(pAnnot);
941     }
942   }
943   SetLock(FALSE);
944 }
945
946 void CPDFSDK_PageView::UpdateRects(CFX_RectArray& rects) {
947   for (int i = 0; i < rects.GetSize(); i++) {
948     CPDF_Rect rc = rects.GetAt(i);
949     CPDFDoc_Environment* pEnv = m_pSDKDoc->GetEnv();
950     pEnv->FFI_Invalidate(m_page, rc.left, rc.top, rc.right, rc.bottom);
951   }
952 }
953
954 void CPDFSDK_PageView::UpdateView(CPDFSDK_Annot* pAnnot) {
955   CPDF_Rect rcWindow = pAnnot->GetRect();
956   CPDFDoc_Environment* pEnv = m_pSDKDoc->GetEnv();
957   pEnv->FFI_Invalidate(m_page, rcWindow.left, rcWindow.top, rcWindow.right,
958                        rcWindow.bottom);
959 }
960
961 int CPDFSDK_PageView::GetPageIndex() {
962   if (m_page) {
963     CPDF_Dictionary* pDic = m_page->m_pFormDict;
964     CPDF_Document* pDoc = m_pSDKDoc->GetDocument();
965     if (pDoc && pDic) {
966       return pDoc->GetPageIndex(pDic->GetObjNum());
967     }
968   }
969   return -1;
970 }
971
972 FX_BOOL CPDFSDK_PageView::IsValidAnnot(void* p) {
973   if (p == NULL)
974     return FALSE;
975   int iCount = m_pAnnotList->Count();
976   for (int i = 0; i < iCount; i++) {
977     if (m_pAnnotList->GetAt(i) == p)
978       return TRUE;
979   }
980   return FALSE;
981 }
982
983 CPDFSDK_Annot* CPDFSDK_PageView::GetFocusAnnot() {
984   CPDFSDK_Annot* pFocusAnnot = m_pSDKDoc->GetFocusAnnot();
985   if (!pFocusAnnot)
986     return NULL;
987
988   for (int i = 0; i < m_fxAnnotArray.GetSize(); i++) {
989     CPDFSDK_Annot* pAnnot = (CPDFSDK_Annot*)m_fxAnnotArray.GetAt(i);
990     if (pAnnot == pFocusAnnot)
991       return pAnnot;
992   }
993   return NULL;
994 }