blob: 1d16e504c7a2e18a8acd67230ffde3dfc648d1ff [file] [log] [blame] [edit]
// Copyright 2014 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 "fpdfsdk/formfiller/cffl_formfield.h"
#include <utility>
#include "constants/form_flags.h"
#include "core/fpdfapi/page/cpdf_page.h"
#include "core/fxcrt/check.h"
#include "core/fxge/cfx_renderdevice.h"
#include "fpdfsdk/cpdfsdk_pageview.h"
#include "fpdfsdk/cpdfsdk_widget.h"
#include "fpdfsdk/formfiller/cffl_perwindowdata.h"
CFFL_FormField::CFFL_FormField(CFFL_InteractiveFormFiller* pFormFiller,
CPDFSDK_Widget* pWidget)
: form_filler_(pFormFiller), widget_(pWidget) {
DCHECK(form_filler_);
}
CFFL_FormField::~CFFL_FormField() {
DestroyWindows();
}
void CFFL_FormField::DestroyWindows() {
while (!maps_.empty()) {
auto it = maps_.begin();
std::unique_ptr<CPWL_Wnd> pWnd = std::move(it->second);
maps_.erase(it);
pWnd->InvalidateProvider(this);
pWnd->Destroy();
}
}
FX_RECT CFFL_FormField::GetViewBBox(const CPDFSDK_PageView* pPageView) {
CPWL_Wnd* pWnd = GetPWLWindow(pPageView);
CFX_FloatRect rcAnnot =
pWnd ? PWLtoFFL(pWnd->GetWindowRect()) : widget_->GetRect();
CFX_FloatRect rcFocus = GetFocusBox(pPageView);
CFX_FloatRect rcWin = rcAnnot;
if (!rcFocus.IsEmpty()) {
rcWin.Union(rcFocus);
}
if (!rcWin.IsEmpty()) {
rcWin.Inflate(1, 1);
rcWin.Normalize();
}
return rcWin.GetOuterRect();
}
void CFFL_FormField::OnDraw(CPDFSDK_PageView* pPageView,
CPDFSDK_Widget* pWidget,
CFX_RenderDevice* pDevice,
const CFX_Matrix& mtUser2Device) {
CPWL_Wnd* pWnd = GetPWLWindow(pPageView);
if (pWnd) {
pWnd->DrawAppearance(pDevice, GetCurMatrix() * mtUser2Device);
return;
}
if (!CFFL_InteractiveFormFiller::IsVisible(pWidget)) {
return;
}
pWidget->DrawAppearance(pDevice, mtUser2Device,
CPDF_Annot::AppearanceMode::kNormal);
}
void CFFL_FormField::OnDrawDeactive(CPDFSDK_PageView* pPageView,
CPDFSDK_Widget* pWidget,
CFX_RenderDevice* pDevice,
const CFX_Matrix& mtUser2Device) {
pWidget->DrawAppearance(pDevice, mtUser2Device,
CPDF_Annot::AppearanceMode::kNormal);
}
void CFFL_FormField::OnMouseEnter(CPDFSDK_PageView* pPageView) {}
void CFFL_FormField::OnMouseExit(CPDFSDK_PageView* pPageView) {
timer_.reset();
DCHECK(widget_);
}
bool CFFL_FormField::OnLButtonDown(CPDFSDK_PageView* pPageView,
CPDFSDK_Widget* pWidget,
Mask<FWL_EVENTFLAG> nFlags,
const CFX_PointF& point) {
CPWL_Wnd* pWnd = CreateOrUpdatePWLWindow(pPageView);
if (!pWnd) {
return false;
}
valid_ = true;
FX_RECT rect = GetViewBBox(pPageView);
InvalidateRect(rect);
if (!rect.Contains(static_cast<int>(point.x), static_cast<int>(point.y))) {
return false;
}
return pWnd->OnLButtonDown(nFlags, FFLtoPWL(point));
}
bool CFFL_FormField::OnLButtonUp(CPDFSDK_PageView* pPageView,
CPDFSDK_Widget* pWidget,
Mask<FWL_EVENTFLAG> nFlags,
const CFX_PointF& point) {
CPWL_Wnd* pWnd = GetPWLWindow(pPageView);
if (!pWnd) {
return false;
}
InvalidateRect(GetViewBBox(pPageView));
pWnd->OnLButtonUp(nFlags, FFLtoPWL(point));
return true;
}
bool CFFL_FormField::OnLButtonDblClk(CPDFSDK_PageView* pPageView,
Mask<FWL_EVENTFLAG> nFlags,
const CFX_PointF& point) {
CPWL_Wnd* pWnd = GetPWLWindow(pPageView);
if (!pWnd) {
return false;
}
pWnd->OnLButtonDblClk(nFlags, FFLtoPWL(point));
return true;
}
bool CFFL_FormField::OnMouseMove(CPDFSDK_PageView* pPageView,
Mask<FWL_EVENTFLAG> nFlags,
const CFX_PointF& point) {
CPWL_Wnd* pWnd = GetPWLWindow(pPageView);
if (!pWnd) {
return false;
}
pWnd->OnMouseMove(nFlags, FFLtoPWL(point));
return true;
}
bool CFFL_FormField::OnMouseWheel(CPDFSDK_PageView* pPageView,
Mask<FWL_EVENTFLAG> nFlags,
const CFX_PointF& point,
const CFX_Vector& delta) {
if (!IsValid()) {
return false;
}
CPWL_Wnd* pWnd = CreateOrUpdatePWLWindow(pPageView);
return pWnd && pWnd->OnMouseWheel(nFlags, FFLtoPWL(point), delta);
}
bool CFFL_FormField::OnRButtonDown(CPDFSDK_PageView* pPageView,
Mask<FWL_EVENTFLAG> nFlags,
const CFX_PointF& point) {
CPWL_Wnd* pWnd = CreateOrUpdatePWLWindow(pPageView);
return pWnd && pWnd->OnRButtonDown(nFlags, FFLtoPWL(point));
}
bool CFFL_FormField::OnRButtonUp(CPDFSDK_PageView* pPageView,
Mask<FWL_EVENTFLAG> nFlags,
const CFX_PointF& point) {
CPWL_Wnd* pWnd = GetPWLWindow(pPageView);
return pWnd && pWnd->OnRButtonUp(nFlags, FFLtoPWL(point));
}
bool CFFL_FormField::OnKeyDown(FWL_VKEYCODE nKeyCode,
Mask<FWL_EVENTFLAG> nFlags) {
if (!IsValid()) {
return false;
}
CPWL_Wnd* pWnd = GetPWLWindow(GetCurPageView());
return pWnd && pWnd->OnKeyDown(nKeyCode, nFlags);
}
bool CFFL_FormField::OnChar(CPDFSDK_Widget* pWidget,
uint32_t nChar,
Mask<FWL_EVENTFLAG> nFlags) {
if (!IsValid()) {
return false;
}
CPWL_Wnd* pWnd = GetPWLWindow(GetCurPageView());
return pWnd && pWnd->OnChar(nChar, nFlags);
}
bool CFFL_FormField::SetIndexSelected(int index, bool selected) {
return false;
}
bool CFFL_FormField::IsIndexSelected(int index) {
return false;
}
WideString CFFL_FormField::GetText() {
if (!IsValid()) {
return WideString();
}
CPWL_Wnd* pWnd = GetPWLWindow(GetCurPageView());
return pWnd ? pWnd->GetText() : WideString();
}
WideString CFFL_FormField::GetSelectedText() {
if (!IsValid()) {
return WideString();
}
CPWL_Wnd* pWnd = GetPWLWindow(GetCurPageView());
return pWnd ? pWnd->GetSelectedText() : WideString();
}
void CFFL_FormField::ReplaceAndKeepSelection(const WideString& text) {
if (!IsValid()) {
return;
}
CPWL_Wnd* pWnd = GetPWLWindow(GetCurPageView());
if (!pWnd) {
return;
}
pWnd->ReplaceAndKeepSelection(text);
}
void CFFL_FormField::ReplaceSelection(const WideString& text) {
if (!IsValid()) {
return;
}
CPWL_Wnd* pWnd = GetPWLWindow(GetCurPageView());
if (!pWnd) {
return;
}
pWnd->ReplaceSelection(text);
}
bool CFFL_FormField::SelectAllText() {
if (!IsValid()) {
return false;
}
CPWL_Wnd* pWnd = GetPWLWindow(GetCurPageView());
return pWnd && pWnd->SelectAllText();
}
bool CFFL_FormField::CanUndo() {
if (!IsValid()) {
return false;
}
CPWL_Wnd* pWnd = GetPWLWindow(GetCurPageView());
return pWnd && pWnd->CanUndo();
}
bool CFFL_FormField::CanRedo() {
if (!IsValid()) {
return false;
}
CPWL_Wnd* pWnd = GetPWLWindow(GetCurPageView());
return pWnd && pWnd->CanRedo();
}
bool CFFL_FormField::Undo() {
if (!IsValid()) {
return false;
}
CPWL_Wnd* pWnd = GetPWLWindow(GetCurPageView());
return pWnd && pWnd->Undo();
}
bool CFFL_FormField::Redo() {
if (!IsValid()) {
return false;
}
CPWL_Wnd* pWnd = GetPWLWindow(GetCurPageView());
return pWnd && pWnd->Redo();
}
void CFFL_FormField::SetFocusForAnnot(CPDFSDK_Widget* pWidget,
Mask<FWL_EVENTFLAG> nFlag) {
CPDFSDK_PageView* pPageView =
form_filler_->GetOrCreatePageView(pWidget->GetPage());
CPWL_Wnd* pWnd = CreateOrUpdatePWLWindow(pPageView);
if (pWnd) {
pWnd->SetFocus();
}
valid_ = true;
InvalidateRect(GetViewBBox(pPageView));
}
void CFFL_FormField::KillFocusForAnnot(Mask<FWL_EVENTFLAG> nFlag) {
if (!IsValid()) {
return;
}
CPDFSDK_PageView* pPageView = form_filler_->GetPageView(widget_->GetPage());
if (!pPageView || !CommitData(pPageView, nFlag)) {
return;
}
if (CPWL_Wnd* pWnd = GetPWLWindow(pPageView)) {
pWnd->KillFocus();
}
bool bDestroyPWLWindow;
switch (widget_->GetFieldType()) {
case FormFieldType::kPushButton:
case FormFieldType::kCheckBox:
case FormFieldType::kRadioButton:
bDestroyPWLWindow = true;
break;
default:
bDestroyPWLWindow = false;
break;
}
EscapeFiller(pPageView, bDestroyPWLWindow);
}
bool CFFL_FormField::IsValid() const {
return valid_;
}
CPWL_Wnd::CreateParams CFFL_FormField::GetCreateParam() {
CPWL_Wnd::CreateParams cp(form_filler_->GetTimerHandler(), form_filler_,
this);
cp.rcRectWnd = GetPDFAnnotRect();
uint32_t dwCreateFlags = PWS_BORDER | PWS_BACKGROUND | PWS_VISIBLE;
uint32_t dwFieldFlag = widget_->GetFieldFlags();
if (dwFieldFlag & pdfium::form_flags::kReadOnly) {
dwCreateFlags |= PWS_READONLY;
}
std::optional<FX_COLORREF> color = widget_->GetFillColor();
if (color.has_value()) {
cp.sBackgroundColor = CFX_Color(color.value());
}
color = widget_->GetBorderColor();
if (color.has_value()) {
cp.sBorderColor = CFX_Color(color.value());
}
cp.sTextColor = CFX_Color(CFX_Color::Type::kGray, 0);
color = widget_->GetTextColor();
if (color.has_value()) {
cp.sTextColor = CFX_Color(color.value());
}
cp.fFontSize = widget_->GetFontSize();
cp.dwBorderWidth = widget_->GetBorderWidth();
cp.nBorderStyle = widget_->GetBorderStyle();
switch (cp.nBorderStyle) {
case BorderStyle::kDash:
cp.sDash = CPWL_Dash{3, 3, 0};
break;
case BorderStyle::kBeveled:
case BorderStyle::kInset:
cp.dwBorderWidth *= 2;
break;
default:
break;
}
if (cp.fFontSize <= 0) {
dwCreateFlags |= PWS_AUTOFONTSIZE;
}
cp.dwFlags = dwCreateFlags;
return cp;
}
CPWL_Wnd* CFFL_FormField::GetPWLWindow(
const CPDFSDK_PageView* pPageView) const {
DCHECK(pPageView);
auto it = maps_.find(pPageView);
return it != maps_.end() ? it->second.get() : nullptr;
}
CPWL_Wnd* CFFL_FormField::CreateOrUpdatePWLWindow(
const CPDFSDK_PageView* pPageView) {
DCHECK(pPageView);
CPWL_Wnd* pWnd = GetPWLWindow(pPageView);
if (!pWnd) {
CPWL_Wnd::CreateParams cp = GetCreateParam();
// TODO(tsepez): maybe pass widget's value age as 4th arg.
auto pPrivateData = std::make_unique<CFFL_PerWindowData>(
widget_, pPageView, widget_->GetAppearanceAge(), 0);
maps_[pPageView] = NewPWLWindow(cp, std::move(pPrivateData));
return maps_[pPageView].get();
}
const auto* pPrivateData =
static_cast<const CFFL_PerWindowData*>(pWnd->GetAttachedData());
if (pPrivateData->AppearanceAgeEquals(widget_->GetAppearanceAge())) {
return pWnd;
}
return ResetPWLWindowForValueAgeInternal(pPageView, widget_,
pPrivateData->GetValueAge());
}
void CFFL_FormField::DestroyPWLWindow(const CPDFSDK_PageView* pPageView) {
auto it = maps_.find(pPageView);
if (it == maps_.end()) {
return;
}
std::unique_ptr<CPWL_Wnd> pWnd = std::move(it->second);
maps_.erase(it);
pWnd->Destroy();
}
CFX_Matrix CFFL_FormField::GetWindowMatrix(
const IPWL_FillerNotify::PerWindowData* pAttached) {
const auto* pPrivateData = static_cast<const CFFL_PerWindowData*>(pAttached);
if (!pPrivateData) {
return CFX_Matrix();
}
const CPDFSDK_PageView* pPageView = pPrivateData->GetPageView();
if (!pPageView) {
return CFX_Matrix();
}
return GetCurMatrix() * pPageView->GetCurrentMatrix();
}
void CFFL_FormField::OnSetFocusForEdit(CPWL_Edit* pEdit) {
// Only sub-classes might have a subordinate edit to focus.
}
CFX_Matrix CFFL_FormField::GetCurMatrix() {
CFX_Matrix mt;
CFX_FloatRect rcDA = widget_->GetPDFAnnot()->GetRect();
switch (widget_->GetRotate()) {
case 90:
mt = CFX_Matrix(0, 1, -1, 0, rcDA.right - rcDA.left, 0);
break;
case 180:
mt = CFX_Matrix(-1, 0, 0, -1, rcDA.right - rcDA.left,
rcDA.top - rcDA.bottom);
break;
case 270:
mt = CFX_Matrix(0, -1, 1, 0, 0, rcDA.top - rcDA.bottom);
break;
case 0:
default:
break;
}
mt.e += rcDA.left;
mt.f += rcDA.bottom;
return mt;
}
CFX_FloatRect CFFL_FormField::GetPDFAnnotRect() const {
CFX_FloatRect rectAnnot = widget_->GetPDFAnnot()->GetRect();
float fWidth = rectAnnot.Width();
float fHeight = rectAnnot.Height();
if ((widget_->GetRotate() / 90) & 0x01) {
std::swap(fWidth, fHeight);
}
return CFX_FloatRect(0, 0, fWidth, fHeight);
}
CPDFSDK_PageView* CFFL_FormField::GetCurPageView() {
return form_filler_->GetOrCreatePageView(widget_->GetPage());
}
CFX_FloatRect CFFL_FormField::GetFocusBox(const CPDFSDK_PageView* pPageView) {
CPWL_Wnd* pWnd = GetPWLWindow(pPageView);
if (!pWnd) {
return CFX_FloatRect();
}
CFX_FloatRect rcFocus = PWLtoFFL(pWnd->GetFocusRect());
return pPageView->GetPDFPage()->GetBBox().Contains(rcFocus) ? rcFocus
: CFX_FloatRect();
}
CFX_FloatRect CFFL_FormField::FFLtoPWL(const CFX_FloatRect& rect) {
return GetCurMatrix().GetInverse().TransformRect(rect);
}
CFX_FloatRect CFFL_FormField::PWLtoFFL(const CFX_FloatRect& rect) {
return GetCurMatrix().TransformRect(rect);
}
CFX_PointF CFFL_FormField::FFLtoPWL(const CFX_PointF& point) {
return GetCurMatrix().GetInverse().Transform(point);
}
CFX_PointF CFFL_FormField::PWLtoFFL(const CFX_PointF& point) {
return GetCurMatrix().Transform(point);
}
bool CFFL_FormField::CommitData(const CPDFSDK_PageView* pPageView,
Mask<FWL_EVENTFLAG> nFlag) {
if (!IsDataChanged(pPageView)) {
return true;
}
ObservedPtr<CPDFSDK_Widget> pObserved(widget_);
if (!form_filler_->OnKeyStrokeCommit(pObserved, pPageView, nFlag)) {
if (!pObserved) {
return false;
}
ResetPWLWindow(pPageView);
return true;
}
if (!pObserved) {
return false;
}
if (!form_filler_->OnValidate(pObserved, pPageView, nFlag)) {
if (!pObserved) {
return false;
}
ResetPWLWindow(pPageView);
return true;
}
if (!pObserved) {
return false;
}
SaveData(pPageView); // may invoking JS to delete this widget.
if (!pObserved) {
return false;
}
form_filler_->OnCalculate(pObserved);
if (!pObserved) {
return false;
}
form_filler_->OnFormat(pObserved);
if (!pObserved) {
return false;
}
return true;
}
bool CFFL_FormField::IsDataChanged(const CPDFSDK_PageView* pPageView) {
return false;
}
void CFFL_FormField::SaveData(const CPDFSDK_PageView* pPageView) {}
#ifdef PDF_ENABLE_XFA
bool CFFL_FormField::IsFieldFull(const CPDFSDK_PageView* pPageView) {
return false;
}
#endif // PDF_ENABLE_XFA
void CFFL_FormField::SetChangeMark() {
form_filler_->OnChange();
}
void CFFL_FormField::GetActionData(const CPDFSDK_PageView* pPageView,
CPDF_AAction::AActionType type,
CFFL_FieldAction& fa) {
fa.sValue = widget_->GetValue();
}
void CFFL_FormField::SetActionData(const CPDFSDK_PageView* pPageView,
CPDF_AAction::AActionType type,
const CFFL_FieldAction& fa) {}
void CFFL_FormField::SavePWLWindowState(const CPDFSDK_PageView* pPageView) {}
void CFFL_FormField::RecreatePWLWindowFromSavedState(
const CPDFSDK_PageView* pPageView) {}
CFFL_PerWindowData* CFFL_FormField::GetPerPWLWindowData(
const CPDFSDK_PageView* pPageView) {
CPWL_Wnd* pWnd = GetPWLWindow(pPageView);
if (!pWnd) {
return nullptr;
}
return static_cast<CFFL_PerWindowData*>(pWnd->GetAttachedData());
}
void CFFL_FormField::ResetPWLWindowForValueAge(
const CPDFSDK_PageView* pPageView,
CPDFSDK_Widget* pWidget,
uint32_t nValueAge) {
// Don't leak PWL_Wnd result to public callers.
ResetPWLWindowForValueAgeInternal(pPageView, pWidget, nValueAge);
}
CPWL_Wnd* CFFL_FormField::ResetPWLWindowForValueAgeInternal(
const CPDFSDK_PageView* pPageView,
CPDFSDK_Widget* pWidget,
uint32_t nValueAge) {
return nValueAge == pWidget->GetValueAge() ? RestorePWLWindow(pPageView)
: ResetPWLWindow(pPageView);
}
CPWL_Wnd* CFFL_FormField::ResetPWLWindow(const CPDFSDK_PageView* pPageView) {
return GetPWLWindow(pPageView);
}
CPWL_Wnd* CFFL_FormField::RestorePWLWindow(const CPDFSDK_PageView* pPageView) {
return GetPWLWindow(pPageView);
}
void CFFL_FormField::OnTimerFired() {}
void CFFL_FormField::EscapeFiller(CPDFSDK_PageView* pPageView,
bool bDestroyPWLWindow) {
valid_ = false;
InvalidateRect(GetViewBBox(pPageView));
if (bDestroyPWLWindow) {
DestroyPWLWindow(pPageView);
}
}
void CFFL_FormField::InvalidateRect(const FX_RECT& rect) {
form_filler_->Invalidate(widget_->GetPage(), rect);
}