1 // Copyright 2014 PDFium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 // Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com
7 #include "../../include/pdfwindow/PDFWindow.h"
8 #include "../../include/pdfwindow/PWL_Wnd.h"
9 #include "../../include/pdfwindow/PWL_ScrollBar.h"
10 #include "../../include/pdfwindow/PWL_Utils.h"
12 #define IsFloatZero(f) ((f) < 0.0001 && (f) > -0.0001)
13 #define IsFloatBigger(fa, fb) ((fa) > (fb) && !IsFloatZero((fa) - (fb)))
14 #define IsFloatSmaller(fa, fb) ((fa) < (fb) && !IsFloatZero((fa) - (fb)))
15 #define IsFloatEqual(fa, fb) IsFloatZero((fa) - (fb))
17 /* ------------------------------- PWL_FLOATRANGE
18 * ------------------------------- */
20 PWL_FLOATRANGE::PWL_FLOATRANGE() {
24 PWL_FLOATRANGE::PWL_FLOATRANGE(FX_FLOAT min, FX_FLOAT max) {
28 void PWL_FLOATRANGE::Default() {
33 void PWL_FLOATRANGE::Set(FX_FLOAT min, FX_FLOAT max) {
43 FX_BOOL PWL_FLOATRANGE::In(FX_FLOAT x) const {
44 return (IsFloatBigger(x, fMin) || IsFloatEqual(x, fMin)) &&
45 (IsFloatSmaller(x, fMax) || IsFloatEqual(x, fMax));
48 FX_FLOAT PWL_FLOATRANGE::GetWidth() const {
52 /* ------------------------------- PWL_SCROLL_PRIVATEDATA
53 * ------------------------------- */
55 PWL_SCROLL_PRIVATEDATA::PWL_SCROLL_PRIVATEDATA() {
59 void PWL_SCROLL_PRIVATEDATA::Default() {
60 ScrollRange.Default();
61 fScrollPos = ScrollRange.fMin;
67 void PWL_SCROLL_PRIVATEDATA::SetScrollRange(FX_FLOAT min, FX_FLOAT max) {
68 ScrollRange.Set(min, max);
70 if (IsFloatSmaller(fScrollPos, ScrollRange.fMin))
71 fScrollPos = ScrollRange.fMin;
72 if (IsFloatBigger(fScrollPos, ScrollRange.fMax))
73 fScrollPos = ScrollRange.fMax;
76 void PWL_SCROLL_PRIVATEDATA::SetClientWidth(FX_FLOAT width) {
80 void PWL_SCROLL_PRIVATEDATA::SetSmallStep(FX_FLOAT step) {
84 void PWL_SCROLL_PRIVATEDATA::SetBigStep(FX_FLOAT step) {
88 FX_BOOL PWL_SCROLL_PRIVATEDATA::SetPos(FX_FLOAT pos) {
89 if (ScrollRange.In(pos)) {
96 void PWL_SCROLL_PRIVATEDATA::AddSmall() {
97 if (!SetPos(fScrollPos + fSmallStep))
98 SetPos(ScrollRange.fMax);
101 void PWL_SCROLL_PRIVATEDATA::SubSmall() {
102 if (!SetPos(fScrollPos - fSmallStep))
103 SetPos(ScrollRange.fMin);
106 void PWL_SCROLL_PRIVATEDATA::AddBig() {
107 if (!SetPos(fScrollPos + fBigStep))
108 SetPos(ScrollRange.fMax);
111 void PWL_SCROLL_PRIVATEDATA::SubBig() {
112 if (!SetPos(fScrollPos - fBigStep))
113 SetPos(ScrollRange.fMin);
116 /* ------------------------------- CPWL_SBButton -------------------------------
119 CPWL_SBButton::CPWL_SBButton(PWL_SCROLLBAR_TYPE eScrollBarType,
120 PWL_SBBUTTON_TYPE eButtonType) {
121 m_eScrollBarType = eScrollBarType;
122 m_eSBButtonType = eButtonType;
124 m_bMouseDown = FALSE;
127 CPWL_SBButton::~CPWL_SBButton() {}
129 CFX_ByteString CPWL_SBButton::GetClassName() const {
130 return "CPWL_SBButton";
133 void CPWL_SBButton::OnCreate(PWL_CREATEPARAM& cp) {
134 cp.eCursorType = FXCT_ARROW;
137 void CPWL_SBButton::GetThisAppearanceStream(CFX_ByteTextBuf& sAppStream) {
138 CPWL_Wnd::GetThisAppearanceStream(sAppStream);
143 CFX_ByteTextBuf sButton;
145 CPDF_Rect rectWnd = GetWindowRect();
147 if (rectWnd.IsEmpty())
152 CPDF_Point ptCenter = GetCenterPoint();
154 switch (m_eScrollBarType) {
156 switch (m_eSBButtonType) {
158 CPDF_Point pt1(ptCenter.x - PWL_TRIANGLE_HALFLEN * 0.5f, ptCenter.y);
159 CPDF_Point pt2(ptCenter.x + PWL_TRIANGLE_HALFLEN * 0.5f,
160 ptCenter.y + PWL_TRIANGLE_HALFLEN);
161 CPDF_Point pt3(ptCenter.x + PWL_TRIANGLE_HALFLEN * 0.5f,
162 ptCenter.y - PWL_TRIANGLE_HALFLEN);
164 if (rectWnd.right - rectWnd.left > PWL_TRIANGLE_HALFLEN * 2 &&
165 rectWnd.top - rectWnd.bottom > PWL_TRIANGLE_HALFLEN) {
167 sButton << pt1.x << " " << pt1.y << " m\n";
168 sButton << pt2.x << " " << pt2.y << " l\n";
169 sButton << pt3.x << " " << pt3.y << " l\n";
170 sButton << pt1.x << " " << pt1.y << " l f\n";
172 sAppStream << sButton;
176 CPDF_Point pt1(ptCenter.x + PWL_TRIANGLE_HALFLEN * 0.5f, ptCenter.y);
177 CPDF_Point pt2(ptCenter.x - PWL_TRIANGLE_HALFLEN * 0.5f,
178 ptCenter.y + PWL_TRIANGLE_HALFLEN);
179 CPDF_Point pt3(ptCenter.x - PWL_TRIANGLE_HALFLEN * 0.5f,
180 ptCenter.y - PWL_TRIANGLE_HALFLEN);
182 if (rectWnd.right - rectWnd.left > PWL_TRIANGLE_HALFLEN * 2 &&
183 rectWnd.top - rectWnd.bottom > PWL_TRIANGLE_HALFLEN) {
185 sButton << pt1.x << " " << pt1.y << " m\n";
186 sButton << pt2.x << " " << pt2.y << " l\n";
187 sButton << pt3.x << " " << pt3.y << " l\n";
188 sButton << pt1.x << " " << pt1.y << " l f\n";
190 sAppStream << sButton;
198 switch (m_eSBButtonType) {
200 CPDF_Point pt1(ptCenter.x - PWL_TRIANGLE_HALFLEN,
201 ptCenter.y - PWL_TRIANGLE_HALFLEN * 0.5f);
202 CPDF_Point pt2(ptCenter.x + PWL_TRIANGLE_HALFLEN,
203 ptCenter.y - PWL_TRIANGLE_HALFLEN * 0.5f);
204 CPDF_Point pt3(ptCenter.x, ptCenter.y + PWL_TRIANGLE_HALFLEN * 0.5f);
206 if (rectWnd.right - rectWnd.left > PWL_TRIANGLE_HALFLEN * 2 &&
207 rectWnd.top - rectWnd.bottom > PWL_TRIANGLE_HALFLEN) {
209 sButton << pt1.x << " " << pt1.y << " m\n";
210 sButton << pt2.x << " " << pt2.y << " l\n";
211 sButton << pt3.x << " " << pt3.y << " l\n";
212 sButton << pt1.x << " " << pt1.y << " l f\n";
214 sAppStream << sButton;
218 CPDF_Point pt1(ptCenter.x - PWL_TRIANGLE_HALFLEN,
219 ptCenter.y + PWL_TRIANGLE_HALFLEN * 0.5f);
220 CPDF_Point pt2(ptCenter.x + PWL_TRIANGLE_HALFLEN,
221 ptCenter.y + PWL_TRIANGLE_HALFLEN * 0.5f);
222 CPDF_Point pt3(ptCenter.x, ptCenter.y - PWL_TRIANGLE_HALFLEN * 0.5f);
224 if (rectWnd.right - rectWnd.left > PWL_TRIANGLE_HALFLEN * 2 &&
225 rectWnd.top - rectWnd.bottom > PWL_TRIANGLE_HALFLEN) {
227 sButton << pt1.x << " " << pt1.y << " m\n";
228 sButton << pt2.x << " " << pt2.y << " l\n";
229 sButton << pt3.x << " " << pt3.y << " l\n";
230 sButton << pt1.x << " " << pt1.y << " l f\n";
232 sAppStream << sButton;
246 void CPWL_SBButton::DrawThisAppearance(CFX_RenderDevice* pDevice,
247 CPDF_Matrix* pUser2Device) {
251 CPDF_Rect rectWnd = GetWindowRect();
252 if (rectWnd.IsEmpty())
255 CPDF_Point ptCenter = GetCenterPoint();
256 int32_t nTransparancy = GetTransparency();
258 switch (m_eScrollBarType) {
260 CPWL_Wnd::DrawThisAppearance(pDevice, pUser2Device);
261 switch (m_eSBButtonType) {
263 CPDF_Point pt1(ptCenter.x - PWL_TRIANGLE_HALFLEN * 0.5f, ptCenter.y);
264 CPDF_Point pt2(ptCenter.x + PWL_TRIANGLE_HALFLEN * 0.5f,
265 ptCenter.y + PWL_TRIANGLE_HALFLEN);
266 CPDF_Point pt3(ptCenter.x + PWL_TRIANGLE_HALFLEN * 0.5f,
267 ptCenter.y - PWL_TRIANGLE_HALFLEN);
269 if (rectWnd.right - rectWnd.left > PWL_TRIANGLE_HALFLEN * 2 &&
270 rectWnd.top - rectWnd.bottom > PWL_TRIANGLE_HALFLEN) {
273 path.SetPointCount(4);
274 path.SetPoint(0, pt1.x, pt1.y, FXPT_MOVETO);
275 path.SetPoint(1, pt2.x, pt2.y, FXPT_LINETO);
276 path.SetPoint(2, pt3.x, pt3.y, FXPT_LINETO);
277 path.SetPoint(3, pt1.x, pt1.y, FXPT_LINETO);
279 pDevice->DrawPath(&path, pUser2Device, NULL,
280 CPWL_Utils::PWLColorToFXColor(
281 PWL_DEFAULT_BLACKCOLOR, nTransparancy),
282 0, FXFILL_ALTERNATE);
286 CPDF_Point pt1(ptCenter.x + PWL_TRIANGLE_HALFLEN * 0.5f, ptCenter.y);
287 CPDF_Point pt2(ptCenter.x - PWL_TRIANGLE_HALFLEN * 0.5f,
288 ptCenter.y + PWL_TRIANGLE_HALFLEN);
289 CPDF_Point pt3(ptCenter.x - PWL_TRIANGLE_HALFLEN * 0.5f,
290 ptCenter.y - PWL_TRIANGLE_HALFLEN);
292 if (rectWnd.right - rectWnd.left > PWL_TRIANGLE_HALFLEN * 2 &&
293 rectWnd.top - rectWnd.bottom > PWL_TRIANGLE_HALFLEN) {
296 path.SetPointCount(4);
297 path.SetPoint(0, pt1.x, pt1.y, FXPT_MOVETO);
298 path.SetPoint(1, pt2.x, pt2.y, FXPT_LINETO);
299 path.SetPoint(2, pt3.x, pt3.y, FXPT_LINETO);
300 path.SetPoint(3, pt1.x, pt1.y, FXPT_LINETO);
302 pDevice->DrawPath(&path, pUser2Device, NULL,
303 CPWL_Utils::PWLColorToFXColor(
304 PWL_DEFAULT_BLACKCOLOR, nTransparancy),
305 0, FXFILL_ALTERNATE);
313 switch (m_eSBButtonType) {
316 CPDF_Rect rcDraw = rectWnd;
317 CPWL_Utils::DrawStrokeRect(pDevice, pUser2Device, rcDraw,
318 ArgbEncode(nTransparancy, 100, 100, 100),
322 rcDraw = CPWL_Utils::DeflateRect(rectWnd, 0.5f);
323 CPWL_Utils::DrawStrokeRect(pDevice, pUser2Device, rcDraw,
324 ArgbEncode(nTransparancy, 255, 255, 255),
329 rcDraw = CPWL_Utils::DeflateRect(rectWnd, 1.0f);
332 CPWL_Utils::DrawShadow(pDevice, pUser2Device, TRUE, FALSE, rcDraw,
333 nTransparancy, 80, 220);
335 CPWL_Utils::DrawFillRect(pDevice, pUser2Device, rcDraw,
336 ArgbEncode(255, 255, 255, 255));
340 if (rectWnd.top - rectWnd.bottom > 6.0f) {
341 FX_FLOAT fX = rectWnd.left + 1.5f;
342 FX_FLOAT fY = rectWnd.bottom;
343 CPDF_Point pts[7] = {CPDF_Point(fX + 2.5f, fY + 4.0f),
344 CPDF_Point(fX + 2.5f, fY + 3.0f),
345 CPDF_Point(fX + 4.5f, fY + 5.0f),
346 CPDF_Point(fX + 6.5f, fY + 3.0f),
347 CPDF_Point(fX + 6.5f, fY + 4.0f),
348 CPDF_Point(fX + 4.5f, fY + 6.0f),
349 CPDF_Point(fX + 2.5f, fY + 4.0f)};
352 CPWL_Utils::DrawFillArea(
353 pDevice, pUser2Device, pts, 7,
354 ArgbEncode(nTransparancy, 255, 255, 255));
356 CPWL_Utils::DrawFillArea(pDevice, pUser2Device, pts, 7,
357 CPWL_Utils::PWLColorToFXColor(
358 PWL_DEFAULT_HEAVYGRAYCOLOR, 255));
363 CPDF_Rect rcDraw = rectWnd;
364 CPWL_Utils::DrawStrokeRect(pDevice, pUser2Device, rcDraw,
365 ArgbEncode(nTransparancy, 100, 100, 100),
369 rcDraw = CPWL_Utils::DeflateRect(rectWnd, 0.5f);
370 CPWL_Utils::DrawStrokeRect(pDevice, pUser2Device, rcDraw,
371 ArgbEncode(nTransparancy, 255, 255, 255),
375 rcDraw = CPWL_Utils::DeflateRect(rectWnd, 1.0f);
377 CPWL_Utils::DrawShadow(pDevice, pUser2Device, TRUE, FALSE, rcDraw,
378 nTransparancy, 80, 220);
380 CPWL_Utils::DrawFillRect(pDevice, pUser2Device, rcDraw,
381 ArgbEncode(255, 255, 255, 255));
385 if (rectWnd.top - rectWnd.bottom > 6.0f) {
386 FX_FLOAT fX = rectWnd.left + 1.5f;
387 FX_FLOAT fY = rectWnd.bottom;
389 CPDF_Point pts[7] = {CPDF_Point(fX + 2.5f, fY + 5.0f),
390 CPDF_Point(fX + 2.5f, fY + 6.0f),
391 CPDF_Point(fX + 4.5f, fY + 4.0f),
392 CPDF_Point(fX + 6.5f, fY + 6.0f),
393 CPDF_Point(fX + 6.5f, fY + 5.0f),
394 CPDF_Point(fX + 4.5f, fY + 3.0f),
395 CPDF_Point(fX + 2.5f, fY + 5.0f)};
398 CPWL_Utils::DrawFillArea(
399 pDevice, pUser2Device, pts, 7,
400 ArgbEncode(nTransparancy, 255, 255, 255));
402 CPWL_Utils::DrawFillArea(pDevice, pUser2Device, pts, 7,
403 CPWL_Utils::PWLColorToFXColor(
404 PWL_DEFAULT_HEAVYGRAYCOLOR, 255));
408 // CPWL_Wnd::DrawThisAppearance(pDevice,pUser2Device);
411 CPDF_Rect rcDraw = rectWnd;
412 CPWL_Utils::DrawStrokeRect(pDevice, pUser2Device, rcDraw,
413 ArgbEncode(nTransparancy, 100, 100, 100),
417 rcDraw = CPWL_Utils::DeflateRect(rectWnd, 0.5f);
418 CPWL_Utils::DrawStrokeRect(pDevice, pUser2Device, rcDraw,
419 ArgbEncode(nTransparancy, 255, 255, 255),
423 // draw shadow effect
425 CPDF_Point ptTop = CPDF_Point(rectWnd.left, rectWnd.top - 1.0f);
426 CPDF_Point ptBottom =
427 CPDF_Point(rectWnd.left, rectWnd.bottom + 1.0f);
432 CPWL_Utils::DrawStrokeLine(pDevice, pUser2Device, ptTop, ptBottom,
433 ArgbEncode(nTransparancy, 210, 210, 210),
439 CPWL_Utils::DrawStrokeLine(pDevice, pUser2Device, ptTop, ptBottom,
440 ArgbEncode(nTransparancy, 220, 220, 220),
446 CPWL_Utils::DrawStrokeLine(pDevice, pUser2Device, ptTop, ptBottom,
447 ArgbEncode(nTransparancy, 240, 240, 240),
453 CPWL_Utils::DrawStrokeLine(pDevice, pUser2Device, ptTop, ptBottom,
454 ArgbEncode(nTransparancy, 240, 240, 240),
460 CPWL_Utils::DrawStrokeLine(pDevice, pUser2Device, ptTop, ptBottom,
461 ArgbEncode(nTransparancy, 210, 210, 210),
467 CPWL_Utils::DrawStrokeLine(pDevice, pUser2Device, ptTop, ptBottom,
468 ArgbEncode(nTransparancy, 180, 180, 180),
474 CPWL_Utils::DrawStrokeLine(pDevice, pUser2Device, ptTop, ptBottom,
475 ArgbEncode(nTransparancy, 150, 150, 150),
481 CPWL_Utils::DrawStrokeLine(pDevice, pUser2Device, ptTop, ptBottom,
482 ArgbEncode(nTransparancy, 150, 150, 150),
488 CPWL_Utils::DrawStrokeLine(pDevice, pUser2Device, ptTop, ptBottom,
489 ArgbEncode(nTransparancy, 180, 180, 180),
495 CPWL_Utils::DrawStrokeLine(pDevice, pUser2Device, ptTop, ptBottom,
496 ArgbEncode(nTransparancy, 210, 210, 210),
499 CPWL_Utils::DrawFillRect(pDevice, pUser2Device, rcDraw,
500 ArgbEncode(255, 255, 255, 255));
505 if (rectWnd.Height() > 8.0f) {
506 FX_COLORREF crStroke = ArgbEncode(nTransparancy, 120, 120, 120);
508 crStroke = CPWL_Utils::PWLColorToFXColor(
509 PWL_DEFAULT_HEAVYGRAYCOLOR, 255);
511 FX_FLOAT nFrictionWidth = 5.0f;
512 FX_FLOAT nFrictionHeight = 5.5f;
515 CPDF_Point(ptCenter.x - nFrictionWidth / 2.0f,
516 ptCenter.y - nFrictionHeight / 2.0f + 0.5f);
518 CPDF_Point(ptCenter.x + nFrictionWidth / 2.0f,
519 ptCenter.y - nFrictionHeight / 2.0f + 0.5f);
521 CPWL_Utils::DrawStrokeLine(pDevice, pUser2Device, ptLeft, ptRight,
527 CPWL_Utils::DrawStrokeLine(pDevice, pUser2Device, ptLeft, ptRight,
533 CPWL_Utils::DrawStrokeLine(pDevice, pUser2Device, ptLeft, ptRight,
540 CPWL_Utils::DrawStrokeLine(pDevice, pUser2Device, ptLeft, ptRight,
541 ArgbEncode(nTransparancy,150,150,150),1.0f);
554 FX_BOOL CPWL_SBButton::OnLButtonDown(const CPDF_Point& point, FX_DWORD nFlag) {
555 CPWL_Wnd::OnLButtonDown(point, nFlag);
557 if (CPWL_Wnd* pParent = GetParentWindow())
558 pParent->OnNotify(this, PNM_LBUTTONDOWN, 0, (intptr_t)&point);
566 FX_BOOL CPWL_SBButton::OnLButtonUp(const CPDF_Point& point, FX_DWORD nFlag) {
567 CPWL_Wnd::OnLButtonUp(point, nFlag);
569 if (CPWL_Wnd* pParent = GetParentWindow())
570 pParent->OnNotify(this, PNM_LBUTTONUP, 0, (intptr_t)&point);
572 m_bMouseDown = FALSE;
578 FX_BOOL CPWL_SBButton::OnMouseMove(const CPDF_Point& point, FX_DWORD nFlag) {
579 CPWL_Wnd::OnMouseMove(point, nFlag);
581 if (CPWL_Wnd* pParent = GetParentWindow()) {
582 pParent->OnNotify(this, PNM_MOUSEMOVE, 0, (intptr_t)&point);
585 if (m_bMouseDown && (m_eSBButtonType == PSBT_MIN || m_eSBButtonType ==
589 (!pParent->OnNotify(this,PNM_LBUTTONDOWN,nFlags,(intptr_t)&point))
598 /* ------------------------------- CPWL_ScrollBar
599 * ---------------------------------- */
601 CPWL_ScrollBar::CPWL_ScrollBar(PWL_SCROLLBAR_TYPE sbType)
608 m_bNotifyForever(TRUE) {}
610 CPWL_ScrollBar::~CPWL_ScrollBar() {}
612 CFX_ByteString CPWL_ScrollBar::GetClassName() const {
613 return "CPWL_ScrollBar";
616 void CPWL_ScrollBar::OnCreate(PWL_CREATEPARAM& cp) {
617 cp.eCursorType = FXCT_ARROW;
620 void CPWL_ScrollBar::RePosChildWnd() {
621 CPDF_Rect rcClient = GetClientRect();
622 CPDF_Rect rcMinButton, rcMaxButton;
623 FX_FLOAT fBWidth = 0;
627 if (rcClient.right - rcClient.left >
628 PWL_SCROLLBAR_BUTTON_WIDTH * 2 + PWL_SCROLLBAR_POSBUTTON_MINWIDTH +
631 CPDF_Rect(rcClient.left, rcClient.bottom,
632 rcClient.left + PWL_SCROLLBAR_BUTTON_WIDTH, rcClient.top);
633 rcMaxButton = CPDF_Rect(rcClient.right - PWL_SCROLLBAR_BUTTON_WIDTH,
634 rcClient.bottom, rcClient.right, rcClient.top);
636 fBWidth = (rcClient.right - rcClient.left -
637 PWL_SCROLLBAR_POSBUTTON_MINWIDTH - 2) /
641 rcMinButton = CPDF_Rect(rcClient.left, rcClient.bottom,
642 rcClient.left + fBWidth, rcClient.top);
643 rcMaxButton = CPDF_Rect(rcClient.right - fBWidth, rcClient.bottom,
644 rcClient.right, rcClient.top);
650 if (IsFloatBigger(rcClient.top - rcClient.bottom,
651 PWL_SCROLLBAR_BUTTON_WIDTH * 2 +
652 PWL_SCROLLBAR_POSBUTTON_MINWIDTH + 2)) {
654 CPDF_Rect(rcClient.left, rcClient.top - PWL_SCROLLBAR_BUTTON_WIDTH,
655 rcClient.right, rcClient.top);
656 rcMaxButton = CPDF_Rect(rcClient.left, rcClient.bottom, rcClient.right,
657 rcClient.bottom + PWL_SCROLLBAR_BUTTON_WIDTH);
659 fBWidth = (rcClient.top - rcClient.bottom -
660 PWL_SCROLLBAR_POSBUTTON_MINWIDTH - 2) /
663 if (IsFloatBigger(fBWidth, 0)) {
664 rcMinButton = CPDF_Rect(rcClient.left, rcClient.top - fBWidth,
665 rcClient.right, rcClient.top);
666 rcMaxButton = CPDF_Rect(rcClient.left, rcClient.bottom,
667 rcClient.right, rcClient.bottom + fBWidth);
675 m_pMinButton->Move(rcMinButton, TRUE, FALSE);
677 m_pMaxButton->Move(rcMaxButton, TRUE, FALSE);
678 MovePosButton(FALSE);
681 void CPWL_ScrollBar::GetThisAppearanceStream(CFX_ByteTextBuf& sAppStream) {
682 CPDF_Rect rectWnd = GetWindowRect();
684 if (IsVisible() && !rectWnd.IsEmpty()) {
685 CFX_ByteTextBuf sButton;
689 << CPWL_Utils::GetColorAppStream(GetBackgroundColor(), TRUE);
690 sButton << rectWnd.left << " " << rectWnd.bottom << " "
691 << rectWnd.right - rectWnd.left << " "
692 << rectWnd.top - rectWnd.bottom << " re b Q\n";
694 sAppStream << sButton;
698 void CPWL_ScrollBar::DrawThisAppearance(CFX_RenderDevice* pDevice,
699 CPDF_Matrix* pUser2Device) {
700 CPDF_Rect rectWnd = GetWindowRect();
702 if (IsVisible() && !rectWnd.IsEmpty()) {
703 CPWL_Utils::DrawFillRect(pDevice, pUser2Device, rectWnd,
704 GetBackgroundColor(), GetTransparency());
706 CPWL_Utils::DrawStrokeLine(
707 pDevice, pUser2Device,
708 CPDF_Point(rectWnd.left + 2.0f, rectWnd.top - 2.0f),
709 CPDF_Point(rectWnd.left + 2.0f, rectWnd.bottom + 2.0f),
710 ArgbEncode(GetTransparency(), 100, 100, 100), 1.0f);
712 CPWL_Utils::DrawStrokeLine(
713 pDevice, pUser2Device,
714 CPDF_Point(rectWnd.right - 2.0f, rectWnd.top - 2.0f),
715 CPDF_Point(rectWnd.right - 2.0f, rectWnd.bottom + 2.0f),
716 ArgbEncode(GetTransparency(), 100, 100, 100), 1.0f);
720 FX_BOOL CPWL_ScrollBar::OnLButtonDown(const CPDF_Point& point, FX_DWORD nFlag) {
721 CPWL_Wnd::OnLButtonDown(point, nFlag);
723 if (HasFlag(PWS_AUTOTRANSPARENT)) {
724 if (GetTransparency() != 255) {
725 SetTransparency(255);
730 CPDF_Rect rcMinArea, rcMaxArea;
732 if (m_pPosButton && m_pPosButton->IsVisible()) {
733 CPDF_Rect rcClient = GetClientRect();
734 CPDF_Rect rcPosButton = m_pPosButton->GetWindowRect();
738 rcMinArea = CPDF_Rect(rcClient.left + PWL_SCROLLBAR_BUTTON_WIDTH,
739 rcClient.bottom, rcPosButton.left, rcClient.top);
740 rcMaxArea = CPDF_Rect(rcPosButton.right, rcClient.bottom,
741 rcClient.right - PWL_SCROLLBAR_BUTTON_WIDTH,
746 rcMinArea = CPDF_Rect(rcClient.left, rcPosButton.top, rcClient.right,
747 rcClient.top - PWL_SCROLLBAR_BUTTON_WIDTH);
748 rcMaxArea = CPDF_Rect(rcClient.left,
749 rcClient.bottom + PWL_SCROLLBAR_BUTTON_WIDTH,
750 rcClient.right, rcPosButton.bottom);
754 rcMinArea.Normalize();
755 rcMaxArea.Normalize();
757 if (rcMinArea.Contains(point.x, point.y)) {
760 NotifyScrollWindow();
763 if (rcMaxArea.Contains(point.x, point.y)) {
766 NotifyScrollWindow();
773 FX_BOOL CPWL_ScrollBar::OnLButtonUp(const CPDF_Point& point, FX_DWORD nFlag) {
774 CPWL_Wnd::OnLButtonUp(point, nFlag);
776 if (HasFlag(PWS_AUTOTRANSPARENT)) {
777 if (GetTransparency() != PWL_SCROLLBAR_TRANSPARANCY) {
778 SetTransparency(PWL_SCROLLBAR_TRANSPARANCY);
784 m_bMouseDown = FALSE;
789 void CPWL_ScrollBar::OnNotify(CPWL_Wnd* pWnd,
793 CPWL_Wnd::OnNotify(pWnd, msg, wParam, lParam);
796 case PNM_LBUTTONDOWN:
797 if (pWnd == m_pMinButton) {
798 OnMinButtonLBDown(*(CPDF_Point*)lParam);
801 if (pWnd == m_pMaxButton) {
802 OnMaxButtonLBDown(*(CPDF_Point*)lParam);
805 if (pWnd == m_pPosButton) {
806 OnPosButtonLBDown(*(CPDF_Point*)lParam);
810 if (pWnd == m_pMinButton) {
811 OnMinButtonLBUp(*(CPDF_Point*)lParam);
814 if (pWnd == m_pMaxButton) {
815 OnMaxButtonLBUp(*(CPDF_Point*)lParam);
818 if (pWnd == m_pPosButton) {
819 OnPosButtonLBUp(*(CPDF_Point*)lParam);
823 if (pWnd == m_pMinButton) {
824 OnMinButtonMouseMove(*(CPDF_Point*)lParam);
827 if (pWnd == m_pMaxButton) {
828 OnMaxButtonMouseMove(*(CPDF_Point*)lParam);
831 if (pWnd == m_pPosButton) {
832 OnPosButtonMouseMove(*(CPDF_Point*)lParam);
835 case PNM_SETSCROLLINFO: {
836 if (PWL_SCROLL_INFO* pInfo = (PWL_SCROLL_INFO*)lParam) {
837 if (FXSYS_memcmp(&m_OriginInfo, pInfo, sizeof(PWL_SCROLL_INFO)) != 0) {
838 m_OriginInfo = *pInfo;
840 pInfo->fContentMax - pInfo->fContentMin - pInfo->fPlateWidth;
841 fMax = fMax > 0.0f ? fMax : 0.0f;
842 SetScrollRange(0, fMax, pInfo->fPlateWidth);
843 SetScrollStep(pInfo->fBigStep, pInfo->fSmallStep);
847 case PNM_SETSCROLLPOS: {
848 FX_FLOAT fPos = *(FX_FLOAT*)lParam;
851 fPos = fPos - m_OriginInfo.fContentMin;
854 fPos = m_OriginInfo.fContentMax - fPos;
862 void CPWL_ScrollBar::CreateButtons(const PWL_CREATEPARAM& cp) {
863 PWL_CREATEPARAM scp = cp;
864 scp.pParentWnd = this;
865 scp.dwBorderWidth = 2;
866 scp.nBorderStyle = PBS_BEVELED;
869 PWS_VISIBLE | PWS_CHILD | PWS_BORDER | PWS_BACKGROUND | PWS_NOREFRESHCLIP;
872 m_pMinButton = new CPWL_SBButton(m_sbType, PSBT_MIN);
873 m_pMinButton->Create(scp);
877 m_pMaxButton = new CPWL_SBButton(m_sbType, PSBT_MAX);
878 m_pMaxButton->Create(scp);
882 m_pPosButton = new CPWL_SBButton(m_sbType, PSBT_POS);
883 m_pPosButton->SetVisible(FALSE);
884 m_pPosButton->Create(scp);
888 FX_FLOAT CPWL_ScrollBar::GetScrollBarWidth() const {
892 return PWL_SCROLLBAR_WIDTH;
895 void CPWL_ScrollBar::SetScrollRange(FX_FLOAT fMin,
897 FX_FLOAT fClientWidth) {
899 m_sData.SetScrollRange(fMin, fMax);
900 m_sData.SetClientWidth(fClientWidth);
902 if (IsFloatSmaller(m_sData.ScrollRange.GetWidth(), 0.0f)) {
903 m_pPosButton->SetVisible(FALSE);
905 m_pPosButton->SetVisible(TRUE);
911 void CPWL_ScrollBar::SetScrollPos(FX_FLOAT fPos) {
912 FX_FLOAT fOldPos = m_sData.fScrollPos;
914 m_sData.SetPos(fPos);
916 if (!IsFloatEqual(m_sData.fScrollPos, fOldPos))
920 void CPWL_ScrollBar::SetScrollStep(FX_FLOAT fBigStep, FX_FLOAT fSmallStep) {
921 m_sData.SetBigStep(fBigStep);
922 m_sData.SetSmallStep(fSmallStep);
925 void CPWL_ScrollBar::MovePosButton(FX_BOOL bRefresh) {
926 ASSERT(m_pPosButton != NULL);
927 ASSERT(m_pMinButton != NULL);
928 ASSERT(m_pMaxButton != NULL);
930 if (m_pPosButton->IsVisible()) {
932 CPDF_Rect rcPosArea, rcPosButton;
934 rcClient = GetClientRect();
935 rcPosArea = GetScrollArea();
937 FX_FLOAT fLeft, fRight, fTop, fBottom;
941 fLeft = TrueToFace(m_sData.fScrollPos);
942 fRight = TrueToFace(m_sData.fScrollPos + m_sData.fClientWidth);
944 if (fRight - fLeft < PWL_SCROLLBAR_POSBUTTON_MINWIDTH)
945 fRight = fLeft + PWL_SCROLLBAR_POSBUTTON_MINWIDTH;
947 if (fRight > rcPosArea.right) {
948 fRight = rcPosArea.right;
949 fLeft = fRight - PWL_SCROLLBAR_POSBUTTON_MINWIDTH;
952 rcPosButton = CPDF_Rect(fLeft, rcPosArea.bottom, fRight, rcPosArea.top);
956 fBottom = TrueToFace(m_sData.fScrollPos + m_sData.fClientWidth);
957 fTop = TrueToFace(m_sData.fScrollPos);
959 if (IsFloatSmaller(fTop - fBottom, PWL_SCROLLBAR_POSBUTTON_MINWIDTH))
960 fBottom = fTop - PWL_SCROLLBAR_POSBUTTON_MINWIDTH;
962 if (IsFloatSmaller(fBottom, rcPosArea.bottom)) {
963 fBottom = rcPosArea.bottom;
964 fTop = fBottom + PWL_SCROLLBAR_POSBUTTON_MINWIDTH;
967 rcPosButton = CPDF_Rect(rcPosArea.left, fBottom, rcPosArea.right, fTop);
972 m_pPosButton->Move(rcPosButton, TRUE, bRefresh);
976 void CPWL_ScrollBar::OnMinButtonLBDown(const CPDF_Point& point) {
979 NotifyScrollWindow();
987 void CPWL_ScrollBar::OnMinButtonLBUp(const CPDF_Point& point) {}
989 void CPWL_ScrollBar::OnMinButtonMouseMove(const CPDF_Point& point) {}
991 void CPWL_ScrollBar::OnMaxButtonLBDown(const CPDF_Point& point) {
994 NotifyScrollWindow();
1002 void CPWL_ScrollBar::OnMaxButtonLBUp(const CPDF_Point& point) {}
1004 void CPWL_ScrollBar::OnMaxButtonMouseMove(const CPDF_Point& point) {}
1006 void CPWL_ScrollBar::OnPosButtonLBDown(const CPDF_Point& point) {
1007 m_bMouseDown = TRUE;
1010 CPDF_Rect rcPosButton = m_pPosButton->GetWindowRect();
1014 m_nOldPos = point.x;
1015 m_fOldPosButton = rcPosButton.left;
1018 m_nOldPos = point.y;
1019 m_fOldPosButton = rcPosButton.top;
1025 void CPWL_ScrollBar::OnPosButtonLBUp(const CPDF_Point& point) {
1027 if (!m_bNotifyForever)
1028 NotifyScrollWindow();
1030 m_bMouseDown = FALSE;
1033 void CPWL_ScrollBar::OnPosButtonMouseMove(const CPDF_Point& point) {
1034 FX_FLOAT fOldScrollPos = m_sData.fScrollPos;
1036 FX_FLOAT fNewPos = 0;
1040 if (FXSYS_fabs(point.x - m_nOldPos) < 1)
1042 fNewPos = FaceToTrue(m_fOldPosButton + point.x - m_nOldPos);
1045 if (FXSYS_fabs(point.y - m_nOldPos) < 1)
1047 fNewPos = FaceToTrue(m_fOldPosButton + point.y - m_nOldPos);
1055 if (IsFloatSmaller(fNewPos, m_sData.ScrollRange.fMin)) {
1056 fNewPos = m_sData.ScrollRange.fMin;
1059 if (IsFloatBigger(fNewPos, m_sData.ScrollRange.fMax)) {
1060 fNewPos = m_sData.ScrollRange.fMax;
1063 m_sData.SetPos(fNewPos);
1068 if (IsFloatSmaller(fNewPos, m_sData.ScrollRange.fMin)) {
1069 fNewPos = m_sData.ScrollRange.fMin;
1072 if (IsFloatBigger(fNewPos, m_sData.ScrollRange.fMax)) {
1073 fNewPos = m_sData.ScrollRange.fMax;
1076 m_sData.SetPos(fNewPos);
1081 if (!IsFloatEqual(fOldScrollPos, m_sData.fScrollPos)) {
1082 MovePosButton(TRUE);
1084 if (m_bNotifyForever)
1085 NotifyScrollWindow();
1090 void CPWL_ScrollBar::NotifyScrollWindow() {
1091 if (CPWL_Wnd* pParent = GetParentWindow()) {
1095 fPos = m_OriginInfo.fContentMin + m_sData.fScrollPos;
1098 fPos = m_OriginInfo.fContentMax - m_sData.fScrollPos;
1101 pParent->OnNotify(this, PNM_SCROLLWINDOW, (intptr_t)m_sbType,
1106 CPDF_Rect CPWL_ScrollBar::GetScrollArea() const {
1107 CPDF_Rect rcClient = GetClientRect();
1110 if (!m_pMinButton || !m_pMaxButton)
1113 CPDF_Rect rcMin = m_pMinButton->GetWindowRect();
1114 CPDF_Rect rcMax = m_pMaxButton->GetWindowRect();
1116 FX_FLOAT fMinWidth = rcMin.right - rcMin.left;
1117 FX_FLOAT fMinHeight = rcMin.top - rcMin.bottom;
1118 FX_FLOAT fMaxWidth = rcMax.right - rcMax.left;
1119 FX_FLOAT fMaxHeight = rcMax.top - rcMax.bottom;
1123 if (rcClient.right - rcClient.left > fMinWidth + fMaxWidth + 2) {
1124 rcArea = CPDF_Rect(rcClient.left + fMinWidth + 1, rcClient.bottom,
1125 rcClient.right - fMaxWidth - 1, rcClient.top);
1127 rcArea = CPDF_Rect(rcClient.left + fMinWidth + 1, rcClient.bottom,
1128 rcClient.left + fMinWidth + 1, rcClient.top);
1132 if (rcClient.top - rcClient.bottom > fMinHeight + fMaxHeight + 2) {
1133 rcArea = CPDF_Rect(rcClient.left, rcClient.bottom + fMinHeight + 1,
1134 rcClient.right, rcClient.top - fMaxHeight - 1);
1136 rcArea = CPDF_Rect(rcClient.left, rcClient.bottom + fMinHeight + 1,
1137 rcClient.right, rcClient.bottom + fMinHeight + 1);
1147 FX_FLOAT CPWL_ScrollBar::TrueToFace(FX_FLOAT fTrue) {
1148 CPDF_Rect rcPosArea;
1149 rcPosArea = GetScrollArea();
1151 FX_FLOAT fFactWidth = m_sData.ScrollRange.GetWidth() + m_sData.fClientWidth;
1152 fFactWidth = fFactWidth == 0 ? 1 : fFactWidth;
1158 fFace = rcPosArea.left +
1159 fTrue * (rcPosArea.right - rcPosArea.left) / fFactWidth;
1162 fFace = rcPosArea.top -
1163 fTrue * (rcPosArea.top - rcPosArea.bottom) / fFactWidth;
1170 FX_FLOAT CPWL_ScrollBar::FaceToTrue(FX_FLOAT fFace) {
1171 CPDF_Rect rcPosArea;
1172 rcPosArea = GetScrollArea();
1174 FX_FLOAT fFactWidth = m_sData.ScrollRange.GetWidth() + m_sData.fClientWidth;
1175 fFactWidth = fFactWidth == 0 ? 1 : fFactWidth;
1181 fTrue = (fFace - rcPosArea.left) * fFactWidth /
1182 (rcPosArea.right - rcPosArea.left);
1185 fTrue = (rcPosArea.top - fFace) * fFactWidth /
1186 (rcPosArea.top - rcPosArea.bottom);
1193 void CPWL_ScrollBar::CreateChildWnd(const PWL_CREATEPARAM& cp) {
1197 void CPWL_ScrollBar::TimerProc() {
1198 PWL_SCROLL_PRIVATEDATA sTemp = m_sData;
1205 if (FXSYS_memcmp(&m_sData, &sTemp, sizeof(PWL_SCROLL_PRIVATEDATA)) != 0) {
1206 MovePosButton(TRUE);
1207 NotifyScrollWindow();