// Copyright 2017 The PDFium Authors // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. // Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com #include "fxjs/cjs_event_context.h" #include "core/fpdfdoc/cpdf_formfield.h" #include "core/fxcrt/autorestorer.h" #include "fpdfsdk/cpdfsdk_formfillenvironment.h" #include "fxjs/cjs_field.h" #include "fxjs/cjs_runtime.h" #include "fxjs/js_define.h" #include "fxjs/js_resources.h" #include "third_party/base/check.h" #include "v8/include/v8-context.h" #include "v8/include/v8-isolate.h" CJS_EventContext::CJS_EventContext(CJS_Runtime* pRuntime) : m_pRuntime(pRuntime), m_pFormFillEnv(pRuntime->GetFormFillEnv()) {} CJS_EventContext::~CJS_EventContext() = default; absl::optional CJS_EventContext::RunScript( const WideString& script) { v8::Isolate::Scope isolate_scope(m_pRuntime->GetIsolate()); v8::HandleScope handle_scope(m_pRuntime->GetIsolate()); v8::Local context = m_pRuntime->GetV8Context(); v8::Context::Scope context_scope(context); if (m_bBusy) { return IJS_Runtime::JS_Error(1, 1, JSGetStringFromID(JSMessage::kBusyError)); } AutoRestorer restorer(&m_bBusy); m_bBusy = true; DCHECK(IsValid()); CJS_Runtime::FieldEvent event(TargetName(), EventKind()); if (!m_pRuntime->AddEventToSet(event)) { return IJS_Runtime::JS_Error( 1, 1, JSGetStringFromID(JSMessage::kDuplicateEventError)); } absl::optional err; if (script.GetLength() > 0) err = m_pRuntime->ExecuteScript(script); m_pRuntime->RemoveEventFromSet(event); Destroy(); return err; } CJS_Field* CJS_EventContext::SourceField() { v8::Local pDocObj = m_pRuntime->NewFXJSBoundObject( CJS_Document::GetObjDefnID(), FXJSOBJTYPE_DYNAMIC); if (pDocObj.IsEmpty()) return nullptr; v8::Local pFieldObj = m_pRuntime->NewFXJSBoundObject( CJS_Field::GetObjDefnID(), FXJSOBJTYPE_DYNAMIC); if (pFieldObj.IsEmpty()) return nullptr; auto* pFormFillEnv = GetFormFillEnv(); auto* pJSDocument = static_cast( CFXJS_Engine::GetObjectPrivate(m_pRuntime->GetIsolate(), pDocObj)); pJSDocument->SetFormFillEnv(pFormFillEnv); auto* pJSField = static_cast( CFXJS_Engine::GetObjectPrivate(m_pRuntime->GetIsolate(), pFieldObj)); pJSField->AttachField(pJSDocument, SourceName()); return pJSField; } CJS_Field* CJS_EventContext::TargetField() { v8::Local pDocObj = m_pRuntime->NewFXJSBoundObject( CJS_Document::GetObjDefnID(), FXJSOBJTYPE_DYNAMIC); if (pDocObj.IsEmpty()) return nullptr; v8::Local pFieldObj = m_pRuntime->NewFXJSBoundObject( CJS_Field::GetObjDefnID(), FXJSOBJTYPE_DYNAMIC); if (pFieldObj.IsEmpty()) return nullptr; auto* pFormFillEnv = GetFormFillEnv(); auto* pJSDocument = static_cast( CFXJS_Engine::GetObjectPrivate(m_pRuntime->GetIsolate(), pDocObj)); pJSDocument->SetFormFillEnv(pFormFillEnv); auto* pJSField = static_cast( CFXJS_Engine::GetObjectPrivate(m_pRuntime->GetIsolate(), pFieldObj)); pJSField->AttachField(pJSDocument, TargetName()); return pJSField; } void CJS_EventContext::OnDoc_Open(const WideString& strTargetName) { Initialize(Kind::kDocOpen); m_strTargetName = strTargetName; } void CJS_EventContext::OnDoc_WillPrint() { Initialize(Kind::kDocWillPrint); } void CJS_EventContext::OnDoc_DidPrint() { Initialize(Kind::kDocDidPrint); } void CJS_EventContext::OnDoc_WillSave() { Initialize(Kind::kDocWillSave); } void CJS_EventContext::OnDoc_DidSave() { Initialize(Kind::kDocDidSave); } void CJS_EventContext::OnDoc_WillClose() { Initialize(Kind::kDocWillClose); } void CJS_EventContext::OnPage_Open() { Initialize(Kind::kPageOpen); } void CJS_EventContext::OnPage_Close() { Initialize(Kind::kPageClose); } void CJS_EventContext::OnPage_InView() { Initialize(Kind::kPageInView); } void CJS_EventContext::OnPage_OutView() { Initialize(Kind::kPageOutView); } void CJS_EventContext::OnField_MouseEnter(bool bModifier, bool bShift, CPDF_FormField* pTarget) { Initialize(Kind::kFieldMouseEnter); m_bModifier = bModifier; m_bShift = bShift; m_strTargetName = pTarget->GetFullName(); } void CJS_EventContext::OnField_MouseExit(bool bModifier, bool bShift, CPDF_FormField* pTarget) { Initialize(Kind::kFieldMouseExit); m_bModifier = bModifier; m_bShift = bShift; m_strTargetName = pTarget->GetFullName(); } void CJS_EventContext::OnField_MouseDown(bool bModifier, bool bShift, CPDF_FormField* pTarget) { Initialize(Kind::kFieldMouseDown); m_bModifier = bModifier; m_bShift = bShift; m_strTargetName = pTarget->GetFullName(); } void CJS_EventContext::OnField_MouseUp(bool bModifier, bool bShift, CPDF_FormField* pTarget) { Initialize(Kind::kFieldMouseUp); m_bModifier = bModifier; m_bShift = bShift; m_strTargetName = pTarget->GetFullName(); } void CJS_EventContext::OnField_Focus(bool bModifier, bool bShift, CPDF_FormField* pTarget, WideString* pValue) { DCHECK(pValue); Initialize(Kind::kFieldFocus); m_bModifier = bModifier; m_bShift = bShift; m_strTargetName = pTarget->GetFullName(); m_pValue = pValue; } void CJS_EventContext::OnField_Blur(bool bModifier, bool bShift, CPDF_FormField* pTarget, WideString* pValue) { DCHECK(pValue); Initialize(Kind::kFieldBlur); m_bModifier = bModifier; m_bShift = bShift; m_strTargetName = pTarget->GetFullName(); m_pValue = pValue; } void CJS_EventContext::OnField_Keystroke(WideString* strChange, const WideString& strChangeEx, bool KeyDown, bool bModifier, int* pSelEnd, int* pSelStart, bool bShift, CPDF_FormField* pTarget, WideString* pValue, bool bWillCommit, bool bFieldFull, bool* pbRc) { DCHECK(pValue); DCHECK(pbRc); DCHECK(pSelStart); DCHECK(pSelEnd); Initialize(Kind::kFieldKeystroke); m_nCommitKey = 0; m_pWideStrChange = strChange; m_WideStrChangeEx = strChangeEx; m_bKeyDown = KeyDown; m_bModifier = bModifier; m_pISelEnd = pSelEnd; m_pISelStart = pSelStart; m_bShift = bShift; m_strTargetName = pTarget->GetFullName(); m_pValue = pValue; m_bWillCommit = bWillCommit; m_pbRc = pbRc; m_bFieldFull = bFieldFull; } void CJS_EventContext::OnField_Validate(WideString* strChange, const WideString& strChangeEx, bool bKeyDown, bool bModifier, bool bShift, CPDF_FormField* pTarget, WideString* pValue, bool* pbRc) { DCHECK(pValue); DCHECK(pbRc); Initialize(Kind::kFieldValidate); m_pWideStrChange = strChange; m_WideStrChangeEx = strChangeEx; m_bKeyDown = bKeyDown; m_bModifier = bModifier; m_bShift = bShift; m_strTargetName = pTarget->GetFullName(); m_pValue = pValue; m_pbRc = pbRc; } void CJS_EventContext::OnField_Calculate(CPDF_FormField* pSource, CPDF_FormField* pTarget, WideString* pValue, bool* pRc) { DCHECK(pValue); DCHECK(pRc); Initialize(Kind::kFieldCalculate); if (pSource) m_strSourceName = pSource->GetFullName(); m_strTargetName = pTarget->GetFullName(); m_pValue = pValue; m_pbRc = pRc; } void CJS_EventContext::OnField_Format(CPDF_FormField* pTarget, WideString* pValue) { DCHECK(pValue); Initialize(Kind::kFieldFormat); m_nCommitKey = 0; m_strTargetName = pTarget->GetFullName(); m_pValue = pValue; m_bWillCommit = true; } void CJS_EventContext::OnExternal_Exec() { Initialize(Kind::kExternalExec); } void CJS_EventContext::Initialize(Kind kind) { m_eKind = kind; m_strTargetName.clear(); m_strSourceName.clear(); m_pWideStrChange = nullptr; m_WideStrChangeDu.clear(); m_WideStrChangeEx.clear(); m_nCommitKey = -1; m_bKeyDown = false; m_bModifier = false; m_bShift = false; m_pISelEnd = nullptr; m_nSelEndDu = 0; m_pISelStart = nullptr; m_nSelStartDu = 0; m_bWillCommit = false; m_pValue = nullptr; m_bFieldFull = false; m_pbRc = nullptr; m_bRcDu = false; m_bValid = true; } void CJS_EventContext::Destroy() { m_bValid = false; } bool CJS_EventContext::IsUserGesture() const { switch (m_eKind) { case Kind::kFieldMouseDown: case Kind::kFieldMouseUp: case Kind::kFieldKeystroke: return true; default: return false; } } WideString& CJS_EventContext::Change() { return m_pWideStrChange ? *m_pWideStrChange : m_WideStrChangeDu; } ByteStringView CJS_EventContext::Name() const { switch (m_eKind) { case Kind::kDocDidPrint: return "DidPrint"; case Kind::kDocDidSave: return "DidSave"; case Kind::kDocOpen: return "Open"; case Kind::kDocWillClose: return "WillClose"; case Kind::kDocWillPrint: return "WillPrint"; case Kind::kDocWillSave: return "WillSave"; case Kind::kExternalExec: return "Exec"; case Kind::kFieldFocus: return "Focus"; case Kind::kFieldBlur: return "Blur"; case Kind::kFieldMouseDown: return "Mouse Down"; case Kind::kFieldMouseUp: return "Mouse Up"; case Kind::kFieldMouseEnter: return "Mouse Enter"; case Kind::kFieldMouseExit: return "Mouse Exit"; case Kind::kFieldCalculate: return "Calculate"; case Kind::kFieldFormat: return "Format"; case Kind::kFieldKeystroke: return "Keystroke"; case Kind::kFieldValidate: return "Validate"; case Kind::kPageOpen: return "Open"; case Kind::kPageClose: return "Close"; case Kind::kPageInView: return "InView"; case Kind::kPageOutView: return "OutView"; default: return ""; } } ByteStringView CJS_EventContext::Type() const { switch (m_eKind) { case Kind::kDocDidPrint: case Kind::kDocDidSave: case Kind::kDocOpen: case Kind::kDocWillClose: case Kind::kDocWillPrint: case Kind::kDocWillSave: return "Doc"; case Kind::kExternalExec: return "External"; case Kind::kFieldBlur: case Kind::kFieldFocus: case Kind::kFieldMouseDown: case Kind::kFieldMouseUp: case Kind::kFieldMouseEnter: case Kind::kFieldMouseExit: case Kind::kFieldCalculate: case Kind::kFieldFormat: case Kind::kFieldKeystroke: case Kind::kFieldValidate: return "Field"; case Kind::kPageOpen: case Kind::kPageClose: case Kind::kPageInView: case Kind::kPageOutView: return "Page"; default: return ""; } } bool& CJS_EventContext::Rc() { return m_pbRc ? *m_pbRc : m_bRcDu; } int CJS_EventContext::SelEnd() const { return m_pISelEnd ? *m_pISelEnd : m_nSelEndDu; } int CJS_EventContext::SelStart() const { return m_pISelStart ? *m_pISelStart : m_nSelStartDu; } void CJS_EventContext::SetSelEnd(int value) { int& target = m_pISelEnd ? *m_pISelEnd : m_nSelEndDu; target = value; } void CJS_EventContext::SetSelStart(int value) { int& target = m_pISelStart ? *m_pISelStart : m_nSelStartDu; target = value; }