253 lines
9.2 KiB
C++
253 lines
9.2 KiB
C++
|
|
// 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/theme/cfwl_checkboxtp.h"
|
||
|
|
|
||
|
|
#include <math.h>
|
||
|
|
|
||
|
|
#include <iterator>
|
||
|
|
|
||
|
|
#include "core/fxge/cfx_path.h"
|
||
|
|
#include "xfa/fde/cfde_textout.h"
|
||
|
|
#include "xfa/fgas/graphics/cfgas_gecolor.h"
|
||
|
|
#include "xfa/fgas/graphics/cfgas_gegraphics.h"
|
||
|
|
#include "xfa/fgas/graphics/cfgas_gepath.h"
|
||
|
|
#include "xfa/fwl/cfwl_checkbox.h"
|
||
|
|
#include "xfa/fwl/cfwl_themebackground.h"
|
||
|
|
#include "xfa/fwl/cfwl_themetext.h"
|
||
|
|
#include "xfa/fwl/cfwl_widget.h"
|
||
|
|
|
||
|
|
namespace {
|
||
|
|
|
||
|
|
constexpr int kSignPath = 100;
|
||
|
|
|
||
|
|
CFX_PointF ScaleBezierPoint(const CFX_PointF& point) {
|
||
|
|
CFX_PointF scaled_point(point);
|
||
|
|
scaled_point.x *= FXSYS_BEZIER;
|
||
|
|
scaled_point.y *= FXSYS_BEZIER;
|
||
|
|
return scaled_point;
|
||
|
|
}
|
||
|
|
|
||
|
|
} // namespace
|
||
|
|
|
||
|
|
CFWL_CheckBoxTP::CFWL_CheckBoxTP() = default;
|
||
|
|
|
||
|
|
CFWL_CheckBoxTP::~CFWL_CheckBoxTP() = default;
|
||
|
|
|
||
|
|
void CFWL_CheckBoxTP::DrawText(const CFWL_ThemeText& pParams) {
|
||
|
|
EnsureTTOInitialized(pParams.GetWidget()->GetThemeProvider());
|
||
|
|
m_pTextOut->SetTextColor(pParams.m_dwStates & CFWL_PartState::kDisabled
|
||
|
|
? FWLTHEME_CAPACITY_TextDisColor
|
||
|
|
: FWLTHEME_CAPACITY_TextColor);
|
||
|
|
CFWL_WidgetTP::DrawText(pParams);
|
||
|
|
}
|
||
|
|
|
||
|
|
void CFWL_CheckBoxTP::DrawSignCheck(CFGAS_GEGraphics* pGraphics,
|
||
|
|
const CFX_RectF& rtSign,
|
||
|
|
FX_ARGB argbFill,
|
||
|
|
const CFX_Matrix& matrix) {
|
||
|
|
EnsureCheckPathInitialized(rtSign.width);
|
||
|
|
DCHECK(m_pCheckPath);
|
||
|
|
|
||
|
|
CFX_Matrix mt;
|
||
|
|
mt.Translate(rtSign.left, rtSign.top);
|
||
|
|
mt.Concat(matrix);
|
||
|
|
|
||
|
|
CFGAS_GEGraphics::StateRestorer restorer(pGraphics);
|
||
|
|
pGraphics->SetFillColor(CFGAS_GEColor(argbFill));
|
||
|
|
pGraphics->FillPath(*m_pCheckPath, CFX_FillRenderOptions::FillType::kWinding,
|
||
|
|
mt);
|
||
|
|
}
|
||
|
|
|
||
|
|
void CFWL_CheckBoxTP::DrawSignCircle(CFGAS_GEGraphics* pGraphics,
|
||
|
|
const CFX_RectF& rtSign,
|
||
|
|
FX_ARGB argbFill,
|
||
|
|
const CFX_Matrix& matrix) {
|
||
|
|
CFGAS_GEPath path;
|
||
|
|
path.AddEllipse(rtSign);
|
||
|
|
|
||
|
|
CFGAS_GEGraphics::StateRestorer restorer(pGraphics);
|
||
|
|
pGraphics->SetFillColor(CFGAS_GEColor(argbFill));
|
||
|
|
pGraphics->FillPath(path, CFX_FillRenderOptions::FillType::kWinding, matrix);
|
||
|
|
}
|
||
|
|
|
||
|
|
void CFWL_CheckBoxTP::DrawSignCross(CFGAS_GEGraphics* pGraphics,
|
||
|
|
const CFX_RectF& rtSign,
|
||
|
|
FX_ARGB argbFill,
|
||
|
|
const CFX_Matrix& matrix) {
|
||
|
|
CFGAS_GEPath path;
|
||
|
|
const float fRight = rtSign.right();
|
||
|
|
const float fBottom = rtSign.bottom();
|
||
|
|
path.AddLine(rtSign.TopLeft(), CFX_PointF(fRight, fBottom));
|
||
|
|
path.AddLine(CFX_PointF(rtSign.left, fBottom),
|
||
|
|
CFX_PointF(fRight, rtSign.top));
|
||
|
|
|
||
|
|
CFGAS_GEGraphics::StateRestorer restorer(pGraphics);
|
||
|
|
pGraphics->SetStrokeColor(CFGAS_GEColor(argbFill));
|
||
|
|
pGraphics->SetLineWidth(1.0f);
|
||
|
|
pGraphics->StrokePath(path, matrix);
|
||
|
|
}
|
||
|
|
|
||
|
|
void CFWL_CheckBoxTP::DrawSignDiamond(CFGAS_GEGraphics* pGraphics,
|
||
|
|
const CFX_RectF& rtSign,
|
||
|
|
FX_ARGB argbFill,
|
||
|
|
const CFX_Matrix& matrix) {
|
||
|
|
CFGAS_GEPath path;
|
||
|
|
const float fWidth = rtSign.width;
|
||
|
|
const float fHeight = rtSign.height;
|
||
|
|
const float fBottom = rtSign.bottom();
|
||
|
|
path.MoveTo(CFX_PointF(rtSign.left + fWidth / 2, rtSign.top));
|
||
|
|
path.LineTo(CFX_PointF(rtSign.left, rtSign.top + fHeight / 2));
|
||
|
|
path.LineTo(CFX_PointF(rtSign.left + fWidth / 2, fBottom));
|
||
|
|
path.LineTo(CFX_PointF(rtSign.right(), rtSign.top + fHeight / 2));
|
||
|
|
path.LineTo(CFX_PointF(rtSign.left + fWidth / 2, rtSign.top));
|
||
|
|
|
||
|
|
CFGAS_GEGraphics::StateRestorer restorer(pGraphics);
|
||
|
|
pGraphics->SetFillColor(CFGAS_GEColor(argbFill));
|
||
|
|
pGraphics->FillPath(path, CFX_FillRenderOptions::FillType::kWinding, matrix);
|
||
|
|
}
|
||
|
|
|
||
|
|
void CFWL_CheckBoxTP::DrawSignSquare(CFGAS_GEGraphics* pGraphics,
|
||
|
|
const CFX_RectF& rtSign,
|
||
|
|
FX_ARGB argbFill,
|
||
|
|
const CFX_Matrix& matrix) {
|
||
|
|
CFGAS_GEPath path;
|
||
|
|
path.AddRectangle(rtSign.left, rtSign.top, rtSign.width, rtSign.height);
|
||
|
|
|
||
|
|
CFGAS_GEGraphics::StateRestorer restorer(pGraphics);
|
||
|
|
pGraphics->SetFillColor(CFGAS_GEColor(argbFill));
|
||
|
|
pGraphics->FillPath(path, CFX_FillRenderOptions::FillType::kWinding, matrix);
|
||
|
|
}
|
||
|
|
|
||
|
|
void CFWL_CheckBoxTP::DrawSignStar(CFGAS_GEGraphics* pGraphics,
|
||
|
|
const CFX_RectF& rtSign,
|
||
|
|
FX_ARGB argbFill,
|
||
|
|
const CFX_Matrix& matrix) {
|
||
|
|
CFGAS_GEPath path;
|
||
|
|
float fBottom = rtSign.bottom();
|
||
|
|
float fRadius = (rtSign.top - fBottom) / (1 + cosf(FXSYS_PI / 5.0f));
|
||
|
|
CFX_PointF ptCenter((rtSign.left + rtSign.right()) / 2.0f,
|
||
|
|
(rtSign.top + fBottom) / 2.0f);
|
||
|
|
|
||
|
|
CFX_PointF points[5];
|
||
|
|
float fAngle = FXSYS_PI / 10.0f;
|
||
|
|
for (auto& point : points) {
|
||
|
|
point =
|
||
|
|
ptCenter + CFX_PointF(fRadius * cosf(fAngle), fRadius * sinf(fAngle));
|
||
|
|
fAngle += FXSYS_PI * 2 / 5.0f;
|
||
|
|
}
|
||
|
|
|
||
|
|
path.MoveTo(points[0]);
|
||
|
|
int next = 0;
|
||
|
|
for (size_t i = 0; i < std::size(points); ++i) {
|
||
|
|
next = (next + 2) % std::size(points);
|
||
|
|
path.LineTo(points[next]);
|
||
|
|
}
|
||
|
|
|
||
|
|
CFGAS_GEGraphics::StateRestorer restorer(pGraphics);
|
||
|
|
pGraphics->SetFillColor(CFGAS_GEColor(argbFill));
|
||
|
|
pGraphics->FillPath(path, CFX_FillRenderOptions::FillType::kWinding, matrix);
|
||
|
|
}
|
||
|
|
|
||
|
|
void CFWL_CheckBoxTP::EnsureCheckPathInitialized(float fCheckLen) {
|
||
|
|
if (m_pCheckPath)
|
||
|
|
return;
|
||
|
|
|
||
|
|
m_pCheckPath = std::make_unique<CFGAS_GEPath>();
|
||
|
|
|
||
|
|
float fWidth = kSignPath;
|
||
|
|
float fHeight = -kSignPath;
|
||
|
|
float fBottom = kSignPath;
|
||
|
|
CFX_PointF pt1(fWidth / 15.0f, fBottom + fHeight * 2 / 5.0f);
|
||
|
|
CFX_PointF pt2(fWidth / 4.5f, fBottom + fHeight / 16.0f);
|
||
|
|
CFX_PointF pt3(fWidth / 3.0f, fBottom);
|
||
|
|
CFX_PointF pt4(fWidth * 14 / 15.0f, fBottom + fHeight * 15 / 16.0f);
|
||
|
|
CFX_PointF pt5(fWidth / 3.6f, fBottom + fHeight / 3.5f);
|
||
|
|
CFX_PointF pt12(fWidth / 7.0f, fBottom + fHeight * 2 / 7.0f);
|
||
|
|
CFX_PointF pt21(fWidth / 5.0f, fBottom + fHeight / 5.0f);
|
||
|
|
CFX_PointF pt23(fWidth / 4.4f, fBottom + fHeight * 0 / 16.0f);
|
||
|
|
CFX_PointF pt32(fWidth / 4.0f, fBottom);
|
||
|
|
CFX_PointF pt34(fWidth * (1 / 7.0f + 7 / 15.0f),
|
||
|
|
fBottom + fHeight * 4 / 5.0f);
|
||
|
|
CFX_PointF pt43(fWidth * (1 / 7.0f + 7 / 15.0f),
|
||
|
|
fBottom + fHeight * 4 / 5.0f);
|
||
|
|
CFX_PointF pt45(fWidth * 7 / 15.0f, fBottom + fHeight * 8 / 7.0f);
|
||
|
|
CFX_PointF pt54(fWidth / 3.4f, fBottom + fHeight / 3.5f);
|
||
|
|
CFX_PointF pt51(fWidth / 3.6f, fBottom + fHeight / 4.0f);
|
||
|
|
CFX_PointF pt15(fWidth / 3.5f, fBottom + fHeight * 3.5f / 5.0f);
|
||
|
|
m_pCheckPath->MoveTo(pt1);
|
||
|
|
|
||
|
|
CFX_PointF p1 = ScaleBezierPoint(pt12 - pt1);
|
||
|
|
CFX_PointF p2 = ScaleBezierPoint(pt21 - pt2);
|
||
|
|
m_pCheckPath->BezierTo(pt1 + p1, pt2 + p2, pt2);
|
||
|
|
|
||
|
|
p1 = ScaleBezierPoint(pt23 - pt2);
|
||
|
|
p2 = ScaleBezierPoint(pt32 - pt3);
|
||
|
|
m_pCheckPath->BezierTo(pt2 + p1, pt3 + p2, pt3);
|
||
|
|
|
||
|
|
p1 = ScaleBezierPoint(pt34 - pt3);
|
||
|
|
p2 = ScaleBezierPoint(pt43 - pt4);
|
||
|
|
m_pCheckPath->BezierTo(pt3 + p1, pt4 + p2, pt4);
|
||
|
|
|
||
|
|
p1 = ScaleBezierPoint(pt45 - pt4);
|
||
|
|
p2 = ScaleBezierPoint(pt54 - pt5);
|
||
|
|
m_pCheckPath->BezierTo(pt4 + p1, pt5 + p2, pt5);
|
||
|
|
|
||
|
|
p1 = ScaleBezierPoint(pt51 - pt5);
|
||
|
|
p2 = ScaleBezierPoint(pt15 - pt1);
|
||
|
|
m_pCheckPath->BezierTo(pt5 + p1, pt1 + p2, pt1);
|
||
|
|
|
||
|
|
float fScale = fCheckLen / kSignPath;
|
||
|
|
CFX_Matrix mt;
|
||
|
|
mt.Scale(fScale, fScale);
|
||
|
|
m_pCheckPath->TransformBy(mt);
|
||
|
|
}
|
||
|
|
|
||
|
|
void CFWL_CheckBoxTP::DrawBackground(const CFWL_ThemeBackground& pParams) {
|
||
|
|
if (pParams.GetPart() != CFWL_ThemePart::Part::kCheckBox)
|
||
|
|
return;
|
||
|
|
|
||
|
|
if ((pParams.m_dwStates & CFWL_PartState::kChecked) ||
|
||
|
|
(pParams.m_dwStates & CFWL_PartState::kNeutral)) {
|
||
|
|
DrawCheckSign(pParams.GetWidget(), pParams.GetGraphics(),
|
||
|
|
pParams.m_PartRect, pParams.m_dwStates, pParams.m_matrix);
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
void CFWL_CheckBoxTP::DrawCheckSign(CFWL_Widget* pWidget,
|
||
|
|
CFGAS_GEGraphics* pGraphics,
|
||
|
|
const CFX_RectF& pRtBox,
|
||
|
|
Mask<CFWL_PartState> iState,
|
||
|
|
const CFX_Matrix& matrix) {
|
||
|
|
CFX_RectF rtSign(pRtBox);
|
||
|
|
uint32_t dwColor =
|
||
|
|
iState & CFWL_PartState::kNeutral ? 0xFFA9A9A9 : 0xFF000000;
|
||
|
|
uint32_t dwStyle = pWidget->GetStyleExts();
|
||
|
|
rtSign.Deflate(rtSign.width / 4, rtSign.height / 4);
|
||
|
|
switch (dwStyle & FWL_STYLEEXT_CKB_SignShapeMask) {
|
||
|
|
case FWL_STYLEEXT_CKB_SignShapeCheck:
|
||
|
|
DrawSignCheck(pGraphics, rtSign, dwColor, matrix);
|
||
|
|
break;
|
||
|
|
case FWL_STYLEEXT_CKB_SignShapeCircle:
|
||
|
|
DrawSignCircle(pGraphics, rtSign, dwColor, matrix);
|
||
|
|
break;
|
||
|
|
case FWL_STYLEEXT_CKB_SignShapeCross:
|
||
|
|
DrawSignCross(pGraphics, rtSign, dwColor, matrix);
|
||
|
|
break;
|
||
|
|
case FWL_STYLEEXT_CKB_SignShapeDiamond:
|
||
|
|
DrawSignDiamond(pGraphics, rtSign, dwColor, matrix);
|
||
|
|
break;
|
||
|
|
case FWL_STYLEEXT_CKB_SignShapeSquare:
|
||
|
|
DrawSignSquare(pGraphics, rtSign, dwColor, matrix);
|
||
|
|
break;
|
||
|
|
case FWL_STYLEEXT_CKB_SignShapeStar:
|
||
|
|
DrawSignStar(pGraphics, rtSign, dwColor, matrix);
|
||
|
|
break;
|
||
|
|
default:
|
||
|
|
break;
|
||
|
|
}
|
||
|
|
}
|