| // Copyright 2014 PDFium Authors. All rights reserved. | 
 | // 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 "core/fxge/cfx_path.h" | 
 | #include "third_party/base/cxx17_backports.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(); | 
 |   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); | 
 |   pGraphics->SaveGraphState(); | 
 |   pGraphics->SetFillColor(CFGAS_GEColor(argbFill)); | 
 |   pGraphics->FillPath(*m_pCheckPath, CFX_FillRenderOptions::FillType::kWinding, | 
 |                       mt); | 
 |   pGraphics->RestoreGraphState(); | 
 | } | 
 |  | 
 | void CFWL_CheckBoxTP::DrawSignCircle(CFGAS_GEGraphics* pGraphics, | 
 |                                      const CFX_RectF& rtSign, | 
 |                                      FX_ARGB argbFill, | 
 |                                      const CFX_Matrix& matrix) { | 
 |   CFGAS_GEPath path; | 
 |   path.AddEllipse(rtSign); | 
 |   pGraphics->SaveGraphState(); | 
 |   pGraphics->SetFillColor(CFGAS_GEColor(argbFill)); | 
 |   pGraphics->FillPath(path, CFX_FillRenderOptions::FillType::kWinding, matrix); | 
 |   pGraphics->RestoreGraphState(); | 
 | } | 
 |  | 
 | void CFWL_CheckBoxTP::DrawSignCross(CFGAS_GEGraphics* pGraphics, | 
 |                                     const CFX_RectF& rtSign, | 
 |                                     FX_ARGB argbFill, | 
 |                                     const CFX_Matrix& matrix) { | 
 |   CFGAS_GEPath path; | 
 |   float fRight = rtSign.right(); | 
 |   float fBottom = rtSign.bottom(); | 
 |   path.AddLine(rtSign.TopLeft(), CFX_PointF(fRight, fBottom)); | 
 |   path.AddLine(CFX_PointF(rtSign.left, fBottom), | 
 |                CFX_PointF(fRight, rtSign.top)); | 
 |  | 
 |   pGraphics->SaveGraphState(); | 
 |   pGraphics->SetStrokeColor(CFGAS_GEColor(argbFill)); | 
 |   pGraphics->SetLineWidth(1.0f); | 
 |   pGraphics->StrokePath(path, matrix); | 
 |   pGraphics->RestoreGraphState(); | 
 | } | 
 |  | 
 | void CFWL_CheckBoxTP::DrawSignDiamond(CFGAS_GEGraphics* pGraphics, | 
 |                                       const CFX_RectF& rtSign, | 
 |                                       FX_ARGB argbFill, | 
 |                                       const CFX_Matrix& matrix) { | 
 |   CFGAS_GEPath path; | 
 |   float fWidth = rtSign.width; | 
 |   float fHeight = rtSign.height; | 
 |   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)); | 
 |  | 
 |   pGraphics->SaveGraphState(); | 
 |   pGraphics->SetFillColor(CFGAS_GEColor(argbFill)); | 
 |   pGraphics->FillPath(path, CFX_FillRenderOptions::FillType::kWinding, matrix); | 
 |   pGraphics->RestoreGraphState(); | 
 | } | 
 |  | 
 | 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); | 
 |   pGraphics->SaveGraphState(); | 
 |   pGraphics->SetFillColor(CFGAS_GEColor(argbFill)); | 
 |   pGraphics->FillPath(path, CFX_FillRenderOptions::FillType::kWinding, matrix); | 
 |   pGraphics->RestoreGraphState(); | 
 | } | 
 |  | 
 | 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 < pdfium::size(points); ++i) { | 
 |     next = (next + 2) % pdfium::size(points); | 
 |     path.LineTo(points[next]); | 
 |   } | 
 |   pGraphics->SaveGraphState(); | 
 |   pGraphics->SetFillColor(CFGAS_GEColor(argbFill)); | 
 |   pGraphics->FillPath(path, CFX_FillRenderOptions::FillType::kWinding, matrix); | 
 |   pGraphics->RestoreGraphState(); | 
 | } | 
 |  | 
 | 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.m_iPart != 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; | 
 |   } | 
 | } |