blob: 8bcedcd7198f933129600201f3da7aeed1af0e71 [file] [log] [blame]
// 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 "xfa/fwl/cfwl_datetimepicker.h"
#include "xfa/fwl/cfwl_app.h"
#include "xfa/fwl/cfwl_event.h"
#include "xfa/fwl/cfwl_eventselectchanged.h"
#include "xfa/fwl/cfwl_messagemouse.h"
#include "xfa/fwl/cfwl_messagesetfocus.h"
#include "xfa/fwl/cfwl_notedriver.h"
#include "xfa/fwl/cfwl_themebackground.h"
#include "xfa/fwl/cfwl_widgetmgr.h"
#include "xfa/fwl/ifwl_themeprovider.h"
namespace pdfium {
namespace {
constexpr int kDateTimePickerHeight = 20;
} // namespace
CFWL_DateTimePicker::CFWL_DateTimePicker(CFWL_App* app)
: CFWL_Widget(app,
Properties{0, FWL_STYLEEXT_DTP_ShortDateFormat, 0},
nullptr),
edit_(cppgc::MakeGarbageCollected<CFWL_DateTimeEdit>(
app->GetHeap()->GetAllocationHandle(),
app,
Properties(),
this)),
month_cal_(cppgc::MakeGarbageCollected<CFWL_MonthCalendar>(
app->GetHeap()->GetAllocationHandle(),
app,
Properties{FWL_STYLE_WGT_Popup | FWL_STYLE_WGT_Border, 0,
FWL_STATE_WGT_Invisible},
this)) {
month_cal_->SetWidgetRect(
CFX_RectF(0, 0, month_cal_->GetAutosizedWidgetRect().Size()));
CFWL_NoteDriver* pNoteDriver = GetFWLApp()->GetNoteDriver();
pNoteDriver->RegisterEventTarget(this, month_cal_);
pNoteDriver->RegisterEventTarget(this, edit_);
}
CFWL_DateTimePicker::~CFWL_DateTimePicker() = default;
void CFWL_DateTimePicker::PreFinalize() {
UnregisterEventTarget();
CFWL_Widget::PreFinalize();
}
void CFWL_DateTimePicker::Trace(cppgc::Visitor* visitor) const {
CFWL_Widget::Trace(visitor);
visitor->Trace(edit_);
visitor->Trace(month_cal_);
}
FWL_Type CFWL_DateTimePicker::GetClassID() const {
return FWL_Type::DateTimePicker;
}
void CFWL_DateTimePicker::Update() {
if (IsLocked())
return;
client_rect_ = GetClientRect();
edit_->SetWidgetRect(client_rect_);
ResetEditAlignment();
edit_->Update();
btn_ = GetThemeProvider()->GetScrollBarWidth();
CFX_RectF rtMonthCal = month_cal_->GetAutosizedWidgetRect();
CFX_RectF rtPopUp(rtMonthCal.left, rtMonthCal.top + kDateTimePickerHeight,
rtMonthCal.width, rtMonthCal.height);
month_cal_->SetWidgetRect(rtPopUp);
month_cal_->Update();
}
FWL_WidgetHit CFWL_DateTimePicker::HitTest(const CFX_PointF& point) {
CFX_RectF rect(0, 0, widget_rect_.width, widget_rect_.height);
if (rect.Contains(point))
return FWL_WidgetHit::Edit;
if (NeedsToShowButton())
rect.width += btn_;
if (rect.Contains(point))
return FWL_WidgetHit::Client;
if (IsMonthCalendarVisible()) {
if (month_cal_->GetWidgetRect().Contains(point)) {
return FWL_WidgetHit::Client;
}
}
return FWL_WidgetHit::Unknown;
}
void CFWL_DateTimePicker::DrawWidget(CFGAS_GEGraphics* pGraphics,
const CFX_Matrix& matrix) {
if (!pGraphics)
return;
if (HasBorder())
DrawBorder(pGraphics, CFWL_ThemePart::Part::kBorder, matrix);
if (!btn_rect_.IsEmpty()) {
DrawDropDownButton(pGraphics, matrix);
}
if (edit_) {
CFX_RectF rtEdit = edit_->GetWidgetRect();
CFX_Matrix mt(1, 0, 0, 1, rtEdit.left, rtEdit.top);
mt.Concat(matrix);
edit_->DrawWidget(pGraphics, mt);
}
if (!IsMonthCalendarVisible())
return;
CFX_RectF rtMonth = month_cal_->GetWidgetRect();
CFX_Matrix mt(1, 0, 0, 1, rtMonth.left, rtMonth.top);
mt.Concat(matrix);
month_cal_->DrawWidget(pGraphics, mt);
}
void CFWL_DateTimePicker::GetCurSel(int32_t& iYear,
int32_t& iMonth,
int32_t& iDay) {
iYear = year_;
iMonth = month_;
iDay = day_;
}
void CFWL_DateTimePicker::SetCurSel(int32_t iYear,
int32_t iMonth,
int32_t iDay) {
if (iYear <= 0 || iYear >= 3000)
return;
if (iMonth <= 0 || iMonth >= 13)
return;
if (iDay <= 0 || iDay >= 32)
return;
year_ = iYear;
month_ = iMonth;
day_ = iDay;
month_cal_->SetSelect(iYear, iMonth, iDay);
}
void CFWL_DateTimePicker::SetEditText(const WideString& wsText) {
if (!edit_) {
return;
}
edit_->SetText(wsText);
RepaintRect(client_rect_);
CFWL_Event ev(CFWL_Event::Type::EditChanged);
DispatchEvent(&ev);
}
WideString CFWL_DateTimePicker::GetEditText() const {
return edit_ ? edit_->GetText() : WideString();
}
size_t CFWL_DateTimePicker::GetEditTextLength() const {
return edit_ ? edit_->GetTextLength() : 0;
}
CFX_RectF CFWL_DateTimePicker::GetBBox() const {
CFX_RectF rect = widget_rect_;
if (NeedsToShowButton())
rect.width += btn_;
if (!IsMonthCalendarVisible())
return rect;
CFX_RectF rtMonth = month_cal_->GetWidgetRect();
rtMonth.Offset(widget_rect_.left, widget_rect_.top);
rect.Union(rtMonth);
return rect;
}
void CFWL_DateTimePicker::ModifyEditStyleExts(uint32_t dwStyleExtsAdded,
uint32_t dwStyleExtsRemoved) {
edit_->ModifyStyleExts(dwStyleExtsAdded, dwStyleExtsRemoved);
}
void CFWL_DateTimePicker::DrawDropDownButton(CFGAS_GEGraphics* pGraphics,
const CFX_Matrix& mtMatrix) {
CFWL_ThemeBackground param(CFWL_ThemePart::Part::kDropDownButton, this,
pGraphics);
param.states_ = btn_state_;
param.part_rect_ = btn_rect_;
param.matrix_ = mtMatrix;
GetThemeProvider()->DrawBackground(param);
}
WideString CFWL_DateTimePicker::FormatDateString(int32_t iYear,
int32_t iMonth,
int32_t iDay) {
if (properties_.style_exts_ & FWL_STYLEEXT_DTP_ShortDateFormat) {
return WideString::Format(L"%d-%d-%d", iYear, iMonth, iDay);
}
return WideString::Format(L"%d Year %d Month %d Day", iYear, iMonth, iDay);
}
void CFWL_DateTimePicker::ShowMonthCalendar() {
if (IsMonthCalendarVisible())
return;
CFX_RectF rtMonthCal = month_cal_->GetAutosizedWidgetRect();
float fPopupMin = rtMonthCal.height;
float fPopupMax = rtMonthCal.height;
CFX_RectF rtAnchor = widget_rect_;
rtAnchor.width = rtMonthCal.width;
rtMonthCal.left = client_rect_.left;
rtMonthCal.top = rtAnchor.Height();
GetPopupPos(fPopupMin, fPopupMax, rtAnchor, &rtMonthCal);
month_cal_->SetWidgetRect(rtMonthCal);
if (year_ > 0 && month_ > 0 && day_ > 0) {
month_cal_->SetSelect(year_, month_, day_);
}
month_cal_->Update();
month_cal_->RemoveStates(FWL_STATE_WGT_Invisible);
CFWL_MessageSetFocus msg(month_cal_);
edit_->GetDelegate()->OnProcessMessage(&msg);
RepaintInflatedMonthCalRect();
}
void CFWL_DateTimePicker::HideMonthCalendar() {
if (!IsMonthCalendarVisible())
return;
month_cal_->SetStates(FWL_STATE_WGT_Invisible);
RepaintInflatedMonthCalRect();
}
void CFWL_DateTimePicker::RepaintInflatedMonthCalRect() {
CFX_RectF rtInvalidate(0, 0, widget_rect_.width, widget_rect_.height);
CFX_RectF rtCal = month_cal_->GetWidgetRect();
rtInvalidate.Union(rtCal);
rtInvalidate.Inflate(2, 2);
RepaintRect(rtInvalidate);
}
bool CFWL_DateTimePicker::IsMonthCalendarVisible() const {
return month_cal_ && month_cal_->IsVisible();
}
void CFWL_DateTimePicker::ResetEditAlignment() {
if (!edit_) {
return;
}
uint32_t dwAdd = 0;
switch (properties_.style_exts_ & FWL_STYLEEXT_DTP_EditHAlignMask) {
case FWL_STYLEEXT_DTP_EditHCenter: {
dwAdd |= FWL_STYLEEXT_EDT_HCenter;
break;
}
case FWL_STYLEEXT_DTP_EditHFar: {
dwAdd |= FWL_STYLEEXT_EDT_HFar;
break;
}
default: {
dwAdd |= FWL_STYLEEXT_EDT_HNear;
break;
}
}
switch (properties_.style_exts_ & FWL_STYLEEXT_DTP_EditVAlignMask) {
case FWL_STYLEEXT_DTP_EditVCenter: {
dwAdd |= FWL_STYLEEXT_EDT_VCenter;
break;
}
case FWL_STYLEEXT_DTP_EditVFar: {
dwAdd |= FWL_STYLEEXT_EDT_VFar;
break;
}
default: {
dwAdd |= FWL_STYLEEXT_EDT_VNear;
break;
}
}
if (properties_.style_exts_ & FWL_STYLEEXT_DTP_EditJustified) {
dwAdd |= FWL_STYLEEXT_EDT_Justified;
}
edit_->ModifyStyleExts(dwAdd, FWL_STYLEEXT_EDT_HAlignMask |
FWL_STYLEEXT_EDT_HAlignModeMask |
FWL_STYLEEXT_EDT_VAlignMask);
}
void CFWL_DateTimePicker::ProcessSelChanged(int32_t iYear,
int32_t iMonth,
int32_t iDay) {
year_ = iYear;
month_ = iMonth;
day_ = iDay;
edit_->SetText(FormatDateString(year_, month_, day_));
edit_->Update();
RepaintRect(client_rect_);
CFWL_EventSelectChanged ev(this, year_, month_, day_);
DispatchEvent(&ev);
}
bool CFWL_DateTimePicker::NeedsToShowButton() const {
return properties_.states_ & FWL_STATE_WGT_Focused ||
month_cal_->GetStates() & FWL_STATE_WGT_Focused ||
edit_->GetStates() & FWL_STATE_WGT_Focused;
}
void CFWL_DateTimePicker::OnProcessMessage(CFWL_Message* pMessage) {
switch (pMessage->GetType()) {
case CFWL_Message::Type::kSetFocus:
OnFocusGained(pMessage);
break;
case CFWL_Message::Type::kKillFocus:
OnFocusLost(pMessage);
break;
case CFWL_Message::Type::kMouse: {
CFWL_MessageMouse* pMouse = static_cast<CFWL_MessageMouse*>(pMessage);
switch (pMouse->cmd_) {
case CFWL_MessageMouse::MouseCommand::kLeftButtonDown:
OnLButtonDown(pMouse);
break;
case CFWL_MessageMouse::MouseCommand::kLeftButtonUp:
OnLButtonUp(pMouse);
break;
case CFWL_MessageMouse::MouseCommand::kMove:
OnMouseMove(pMouse);
break;
case CFWL_MessageMouse::MouseCommand::kLeave:
OnMouseLeave(pMouse);
break;
default:
break;
}
break;
}
case CFWL_Message::Type::kKey: {
if (edit_->GetStates() & FWL_STATE_WGT_Focused) {
edit_->GetDelegate()->OnProcessMessage(pMessage);
return;
}
break;
}
default:
break;
}
// Dst target could be |this|, continue only if not destroyed by above.
if (pMessage->GetDstTarget())
CFWL_Widget::OnProcessMessage(pMessage);
}
void CFWL_DateTimePicker::OnDrawWidget(CFGAS_GEGraphics* pGraphics,
const CFX_Matrix& matrix) {
DrawWidget(pGraphics, matrix);
}
void CFWL_DateTimePicker::OnFocusGained(CFWL_Message* pMsg) {
properties_.states_ |= FWL_STATE_WGT_Focused;
if (edit_ && !(edit_->GetStyleExts() & FWL_STYLEEXT_EDT_ReadOnly)) {
btn_rect_ = CFX_RectF(widget_rect_.width, 0, btn_, widget_rect_.height - 1);
}
CFX_RectF rtInvalidate(btn_rect_);
pMsg->SetDstTarget(edit_);
edit_->GetDelegate()->OnProcessMessage(pMsg);
rtInvalidate.Inflate(2, 2);
RepaintRect(rtInvalidate);
}
void CFWL_DateTimePicker::OnFocusLost(CFWL_Message* pMsg) {
CFX_RectF rtInvalidate(btn_rect_);
properties_.states_ &= ~FWL_STATE_WGT_Focused;
btn_rect_ = CFX_RectF();
HideMonthCalendar();
if (edit_->GetStates() & FWL_STATE_WGT_Focused) {
edit_->GetDelegate()->OnProcessMessage(pMsg);
}
rtInvalidate.Inflate(2, 2);
RepaintRect(rtInvalidate);
}
void CFWL_DateTimePicker::OnLButtonDown(CFWL_MessageMouse* pMsg) {
if (!pMsg)
return;
if (!btn_rect_.Contains(pMsg->pos_)) {
return;
}
if (IsMonthCalendarVisible()) {
HideMonthCalendar();
return;
}
ShowMonthCalendar();
lbtn_down_ = true;
RepaintRect(client_rect_);
}
void CFWL_DateTimePicker::OnLButtonUp(CFWL_MessageMouse* pMsg) {
if (!pMsg)
return;
lbtn_down_ = false;
if (btn_rect_.Contains(pMsg->pos_)) {
btn_state_ = CFWL_PartState::kHovered;
} else {
btn_state_ = CFWL_PartState::kNormal;
}
RepaintRect(btn_rect_);
}
void CFWL_DateTimePicker::OnMouseMove(CFWL_MessageMouse* pMsg) {
if (!btn_rect_.Contains(pMsg->pos_)) {
btn_state_ = CFWL_PartState::kNormal;
}
RepaintRect(btn_rect_);
}
void CFWL_DateTimePicker::OnMouseLeave(CFWL_MessageMouse* pMsg) {
if (!pMsg)
return;
btn_state_ = CFWL_PartState::kNormal;
RepaintRect(btn_rect_);
}
void CFWL_DateTimePicker::GetPopupPos(float fMinHeight,
float fMaxHeight,
const CFX_RectF& rtAnchor,
CFX_RectF* pPopupRect) {
GetWidgetMgr()->GetAdapterPopupPos(this, fMinHeight, fMaxHeight, rtAnchor,
pPopupRect);
}
void CFWL_DateTimePicker::ClearText() {
edit_->ClearText();
}
void CFWL_DateTimePicker::SelectAll() {
edit_->SelectAll();
}
void CFWL_DateTimePicker::ClearSelection() {
edit_->ClearSelection();
}
std::optional<WideString> CFWL_DateTimePicker::Copy() {
return edit_->Copy();
}
std::optional<WideString> CFWL_DateTimePicker::Cut() {
return edit_->Cut();
}
bool CFWL_DateTimePicker::Paste(const WideString& wsPaste) {
return edit_->Paste(wsPaste);
}
bool CFWL_DateTimePicker::Undo() {
return edit_->Undo();
}
bool CFWL_DateTimePicker::Redo() {
return edit_->Redo();
}
bool CFWL_DateTimePicker::CanUndo() {
return edit_->CanUndo();
}
bool CFWL_DateTimePicker::CanRedo() {
return edit_->CanRedo();
}
} // namespace pdfium