1 /* 2 * PROJECT: ReactOS UI Layout Engine 3 * LICENSE: GPL-2.0+ (https://spdx.org/licenses/GPL-2.0+) 4 * FILE: base/applications/rapps/include/rosui.h 5 * PURPOSE: ATL Layout engine for RAPPS 6 * COPYRIGHT: Copyright 2015 David Quintana (gigaherz@gmail.com) 7 */ 8 #pragma once 9 10 #include <atlwin.h> 11 12 template<class T, INT GrowthRate = 10> 13 class CPointerArray 14 { 15 protected: 16 HDPA m_hDpa; 17 18 public: 19 CPointerArray() 20 { 21 m_hDpa = DPA_Create(GrowthRate); 22 } 23 24 ~CPointerArray() 25 { 26 DPA_DestroyCallback(m_hDpa, s_OnRemoveItem, this); 27 } 28 29 private: 30 static INT CALLBACK s_OnRemoveItem(PVOID ptr, PVOID context) 31 { 32 CPointerArray * self = (CPointerArray*) context; 33 return (INT) self->OnRemoveItem(reinterpret_cast<T*>(ptr)); 34 } 35 36 static INT CALLBACK s_OnCompareItems(PVOID p1, PVOID p2, LPARAM lParam) 37 { 38 CPointerArray * self = (CPointerArray*) lParam; 39 return self->OnCompareItems(reinterpret_cast<T*>(p1), reinterpret_cast<T*>(p2)); 40 } 41 42 public: 43 virtual BOOL OnRemoveItem(T * ptr) 44 { 45 return TRUE; 46 } 47 48 virtual INT OnCompareItems(T * p1, T * p2) 49 { 50 INT_PTR t = (reinterpret_cast<INT_PTR>(p2) - reinterpret_cast<INT_PTR>(p1)); 51 if (t > 0) 52 return 1; 53 if (t < 0) 54 return -1; 55 return 0; 56 } 57 58 public: 59 INT GetCount() const 60 { 61 return DPA_GetPtrCount(m_hDpa); 62 } 63 64 T* Get(INT i) const 65 { 66 return (T*) DPA_GetPtr(m_hDpa, i); 67 } 68 69 BOOL Set(INT i, T* ptr) 70 { 71 return DPA_SetPtr(m_hDpa, i, ptr); 72 } 73 74 INT Insert(INT at, T* ptr) 75 { 76 return DPA_InsertPtr(m_hDpa, at, ptr); 77 } 78 79 INT Append(T* ptr) 80 { 81 return DPA_InsertPtr(m_hDpa, DA_LAST, ptr); 82 } 83 84 INT IndexOf(T* ptr) const 85 { 86 return DPA_GetPtrIndex(m_hDpa, ptr); 87 } 88 89 BOOL Remove(T* ptr) 90 { 91 INT i = IndexOf(ptr); 92 if (i < 0) 93 return FALSE; 94 return RemoveAt(i); 95 } 96 97 BOOL RemoveAt(INT i) 98 { 99 T* ptr = (T*) DPA_GetPtr(m_hDpa, i); 100 OnRemoveItem(ptr); 101 return DPA_DeletePtr(m_hDpa, i); 102 } 103 104 BOOL Clear() 105 { 106 DPA_EnumCallback(s_OnRemoveItem, this); 107 return DPA_DeleteAllPtrs(m_hDpa); 108 } 109 110 BOOL Sort() 111 { 112 return DPA_Sort(m_hDpa, s_OnCompareItems, (LPARAM)this); 113 } 114 115 INT Search(T* item, INT iStart, UINT uFlags) 116 { 117 return DPA_Search(m_hDpa, item, 0, s_OnCompareItems, (LPARAM)this, 0); 118 } 119 }; 120 121 class CUiRect 122 : public RECT 123 { 124 public: 125 CUiRect() 126 { 127 left = right = top = bottom = 0; 128 } 129 130 CUiRect(INT l, INT t, INT r, INT b) 131 { 132 left = l; 133 right = r; 134 top = t; 135 bottom = b; 136 } 137 }; 138 139 class CUiMargin 140 : public CUiRect 141 { 142 public: 143 CUiMargin() 144 { 145 } 146 147 CUiMargin(INT all) 148 : CUiRect(all, all, all, all) 149 { 150 } 151 152 CUiMargin(INT horz, INT vert) 153 : CUiRect(horz, vert, horz, vert) 154 { 155 } 156 }; 157 158 class CUiMeasure 159 { 160 public: 161 enum MeasureType 162 { 163 Type_FitContent = 0, 164 Type_Fixed = 1, 165 Type_Percent = 2, 166 Type_FitParent = 3 167 }; 168 169 private: 170 MeasureType m_Type; 171 INT m_Value; 172 173 public: 174 CUiMeasure() 175 { 176 m_Type = Type_FitContent; 177 m_Value = 0; 178 } 179 180 CUiMeasure(MeasureType type, INT value) 181 { 182 m_Type = type; 183 m_Value = value; 184 } 185 186 INT ComputeMeasure(INT parent, INT content) 187 { 188 switch (m_Type) 189 { 190 case Type_FitContent: 191 return content; 192 case Type_Fixed: 193 return m_Value; 194 case Type_Percent: 195 return max(content, parent * m_Value / 100); 196 case Type_FitParent: 197 return parent; 198 } 199 200 return 0; 201 } 202 203 public: 204 static CUiMeasure FitContent() 205 { 206 return CUiMeasure(Type_FitContent, 0); 207 } 208 209 static CUiMeasure FitParent() 210 { 211 return CUiMeasure(Type_FitParent, 0); 212 } 213 214 static CUiMeasure Fixed(INT pixels) 215 { 216 return CUiMeasure(Type_Fixed, pixels); 217 } 218 219 static CUiMeasure Percent(INT percent) 220 { 221 return CUiMeasure(Type_Percent, percent); 222 } 223 }; 224 225 enum CUiAlignment 226 { 227 UiAlign_LeftTop, 228 UiAlign_Middle, 229 UiAlign_RightBtm, 230 UiAlign_Stretch 231 }; 232 233 class CUiBox 234 { 235 public: 236 CUiMargin m_Margin; 237 238 CUiAlignment m_HorizontalAlignment; 239 CUiAlignment m_VerticalAlignment; 240 241 protected: 242 CUiBox() 243 { 244 m_HorizontalAlignment = UiAlign_LeftTop; 245 m_VerticalAlignment = UiAlign_LeftTop; 246 } 247 248 virtual VOID ComputeRect(RECT parentRect, RECT currentRect, RECT* newRect) 249 { 250 parentRect.left += m_Margin.left; 251 parentRect.right -= m_Margin.right; 252 parentRect.top += m_Margin.top; 253 parentRect.bottom -= m_Margin.bottom; 254 255 if (parentRect.right < parentRect.left) 256 parentRect.right = parentRect.left; 257 258 if (parentRect.bottom < parentRect.top) 259 parentRect.bottom = parentRect.top; 260 261 SIZE szParent = {parentRect.right - parentRect.left, parentRect.bottom - parentRect.top}; 262 SIZE szCurrent = {currentRect.right - currentRect.left, currentRect.bottom - currentRect.top}; 263 264 currentRect = parentRect; 265 266 switch (m_HorizontalAlignment) 267 { 268 case UiAlign_LeftTop: 269 currentRect.right = currentRect.left + szCurrent.cx; 270 break; 271 case UiAlign_Middle: 272 currentRect.left = parentRect.left + (szParent.cx - szCurrent.cx) / 2; 273 currentRect.right = currentRect.left + szCurrent.cx; 274 break; 275 case UiAlign_RightBtm: 276 currentRect.left = currentRect.right - szCurrent.cx; 277 break; 278 default: 279 break; 280 } 281 282 switch (m_VerticalAlignment) 283 { 284 case UiAlign_LeftTop: 285 currentRect.bottom = currentRect.top + szCurrent.cy; 286 break; 287 case UiAlign_Middle: 288 currentRect.top = parentRect.top + (szParent.cy - szCurrent.cy) / 2; 289 currentRect.bottom = currentRect.top + szCurrent.cy; 290 break; 291 case UiAlign_RightBtm: 292 currentRect.top = currentRect.bottom - szCurrent.cy; 293 break; 294 default: 295 break; 296 } 297 298 *newRect = currentRect; 299 } 300 301 302 public: 303 virtual VOID ComputeMinimalSize(SIZE* size) 304 { 305 // Override in subclass 306 size->cx = max(size->cx, 0); 307 size->cy = min(size->cy, 0); 308 }; 309 310 virtual VOID ComputeContentBounds(RECT* rect) 311 { 312 // Override in subclass 313 }; 314 315 virtual DWORD_PTR CountSizableChildren() 316 { 317 // Override in subclass 318 return 0; 319 }; 320 321 virtual HDWP OnParentSize(RECT parentRect, HDWP hDwp) 322 { 323 // Override in subclass 324 return NULL; 325 }; 326 }; 327 328 class CUiPrimitive 329 { 330 protected: 331 CUiPrimitive * m_Parent; 332 333 public: 334 virtual ~CUiPrimitive() {} 335 336 virtual CUiBox * AsBox() { return NULL; } 337 }; 338 339 class CUiCollection : 340 public CPointerArray < CUiPrimitive > 341 { 342 virtual BOOL OnRemoveItem(CUiPrimitive * ptr) 343 { 344 delete ptr; 345 return TRUE; 346 } 347 }; 348 349 class CUiContainer 350 { 351 protected: 352 CUiCollection m_Children; 353 354 public: 355 CUiCollection& Children() { return m_Children; } 356 }; 357 358 class CUiPanel : 359 public CUiPrimitive, 360 public CUiBox, 361 public CUiContainer 362 { 363 public: 364 CUiMeasure m_Width; 365 CUiMeasure m_Height; 366 367 CUiPanel() 368 { 369 m_Width = CUiMeasure::FitParent(); 370 m_Height = CUiMeasure::FitParent(); 371 } 372 373 virtual ~CUiPanel() 374 { 375 } 376 377 virtual CUiBox * AsBox() { return this; } 378 379 virtual VOID ComputeMinimalSize(SIZE* size) 380 { 381 for (INT i = 0; i < m_Children.GetCount(); i++) 382 { 383 CUiBox * box = m_Children.Get(i)->AsBox(); 384 if (box) 385 { 386 box->ComputeMinimalSize(size); 387 } 388 } 389 }; 390 391 virtual VOID ComputeContentBounds(RECT* rect) 392 { 393 for (INT i = 0; i < m_Children.GetCount(); i++) 394 { 395 CUiBox * box = m_Children.Get(i)->AsBox(); 396 if (box) 397 { 398 box->ComputeContentBounds(rect); 399 } 400 } 401 }; 402 403 virtual DWORD_PTR CountSizableChildren() 404 { 405 INT count = 0; 406 for (INT i = 0; i < m_Children.GetCount(); i++) 407 { 408 CUiBox * box = m_Children.Get(i)->AsBox(); 409 if (box) 410 { 411 count += box->CountSizableChildren(); 412 } 413 } 414 return count; 415 } 416 417 virtual HDWP OnParentSize(RECT parentRect, HDWP hDwp) 418 { 419 RECT rect = {0}; 420 421 SIZE content = {0}; 422 ComputeMinimalSize(&content); 423 424 INT preferredWidth = m_Width.ComputeMeasure(parentRect.right - parentRect.left, content.cx); 425 INT preferredHeight = m_Height.ComputeMeasure(parentRect.bottom - parentRect.top, content.cy); 426 427 rect.right = preferredWidth; 428 rect.bottom = preferredHeight; 429 430 ComputeRect(parentRect, rect, &rect); 431 432 for (INT i = 0; i < m_Children.GetCount(); i++) 433 { 434 CUiBox * box = m_Children.Get(i)->AsBox(); 435 if (box) 436 { 437 hDwp = box->OnParentSize(rect, hDwp); 438 } 439 } 440 441 return hDwp; 442 } 443 }; 444 445 template<class T = CWindow> 446 class CUiWindow : 447 public CUiPrimitive, 448 public CUiBox, 449 public T 450 { 451 public: 452 virtual CUiBox * AsBox() { return this; } 453 454 HWND GetWindow() { return T::m_hWnd; } 455 456 virtual VOID ComputeMinimalSize(SIZE* size) 457 { 458 // TODO: Maybe use WM_GETMINMAXINFO? 459 return CUiBox::ComputeMinimalSize(size); 460 }; 461 462 virtual VOID ComputeContentBounds(RECT* rect) 463 { 464 RECT r; 465 ::GetWindowRect(T::m_hWnd, &r); 466 rect->left = min(rect->left, r.left); 467 rect->top = min(rect->top, r.top); 468 rect->right = max(rect->right, r.right); 469 rect->bottom = max(rect->bottom, r.bottom); 470 }; 471 472 virtual DWORD_PTR CountSizableChildren() 473 { 474 return 1; 475 }; 476 477 virtual HDWP OnParentSize(RECT parentRect, HDWP hDwp) 478 { 479 RECT rect; 480 481 ::GetWindowRect(T::m_hWnd, &rect); 482 483 ComputeRect(parentRect, rect, &rect); 484 485 if (hDwp) 486 { 487 return ::DeferWindowPos(hDwp, T::m_hWnd, NULL, rect.left, rect.top, rect.right - rect.left, rect.bottom - rect.top, SWP_NOACTIVATE | SWP_NOZORDER); 488 } 489 else 490 { 491 T::SetWindowPos(NULL, rect.left, rect.top, rect.right - rect.left, rect.bottom - rect.top, SWP_NOACTIVATE | SWP_NOZORDER | SWP_DEFERERASE); 492 return NULL; 493 } 494 }; 495 496 virtual ~CUiWindow() 497 { 498 T::DestroyWindow(); 499 } 500 501 VOID GetWindowTextW(ATL::CStringW& szText) 502 { 503 INT length = CWindow::GetWindowTextLengthW() + 1; 504 CWindow::GetWindowTextW(szText.GetBuffer(length), length); 505 szText.ReleaseBuffer(); 506 } 507 }; 508 509 class CUiSplitPanel : 510 public CUiPrimitive, 511 public CUiBox, 512 public CWindowImpl<CUiSplitPanel> 513 { 514 static const INT THICKNESS = 4; 515 516 protected: 517 518 HCURSOR m_hCursor; 519 520 CUiPanel m_First; 521 CUiPanel m_Second; 522 523 RECT m_LastRect; 524 525 BOOL m_HasOldRect; 526 527 public: 528 INT m_Pos; 529 BOOL m_Horizontal; 530 BOOL m_DynamicFirst; 531 INT m_MinFirst; 532 INT m_MinSecond; 533 534 CUiMeasure m_Width; 535 CUiMeasure m_Height; 536 537 CUiSplitPanel() 538 { 539 m_Width = CUiMeasure::FitParent(); 540 m_Height = CUiMeasure::FitParent(); 541 m_Pos = 100; 542 m_MinFirst = 100; 543 m_MinSecond = 100; 544 m_DynamicFirst = FALSE; 545 m_HasOldRect = FALSE; 546 } 547 548 virtual ~CUiSplitPanel() 549 { 550 } 551 552 virtual CUiBox * AsBox() { return this; } 553 554 CUiCollection& First() { return m_First.Children(); } 555 CUiCollection& Second() { return m_Second.Children(); } 556 557 virtual VOID ComputeMinimalSize(SIZE* size) 558 { 559 if (m_Horizontal) 560 size->cx = max(size->cx, THICKNESS); 561 else 562 size->cy = max(size->cy, THICKNESS); 563 m_First.ComputeMinimalSize(size); 564 m_Second.ComputeMinimalSize(size); 565 }; 566 567 virtual VOID ComputeContentBounds(RECT* rect) 568 { 569 RECT r; 570 571 m_First.ComputeContentBounds(rect); 572 m_Second.ComputeContentBounds(rect); 573 574 ::GetWindowRect(m_hWnd, &r); 575 576 rect->left = min(rect->left, r.left); 577 rect->top = min(rect->top, r.top); 578 rect->right = max(rect->right, r.right); 579 rect->bottom = max(rect->bottom, r.bottom); 580 }; 581 582 virtual DWORD_PTR CountSizableChildren() 583 { 584 INT count = 1; 585 count += m_First.CountSizableChildren(); 586 count += m_Second.CountSizableChildren(); 587 return count; 588 }; 589 590 virtual HDWP OnParentSize(RECT parentRect, HDWP hDwp) 591 { 592 RECT rect = {0}; 593 594 SIZE content = {0}; 595 ComputeMinimalSize(&content); 596 597 INT preferredWidth = m_Width.ComputeMeasure(parentRect.right - parentRect.left, content.cx); 598 INT preferredHeight = m_Width.ComputeMeasure(parentRect.bottom - parentRect.top, content.cy); 599 600 rect.right = preferredWidth; 601 rect.bottom = preferredHeight; 602 603 ComputeRect(parentRect, rect, &rect); 604 605 SIZE growth = {0}; 606 if (m_HasOldRect) 607 { 608 RECT oldRect = m_LastRect; 609 610 growth.cx = (parentRect.right - parentRect.left) - (oldRect.right - oldRect.left); 611 growth.cy = (parentRect.bottom - parentRect.top) - (oldRect.bottom - oldRect.top); 612 } 613 614 RECT splitter = rect; 615 RECT first = rect; 616 RECT second = rect; 617 618 if (m_Horizontal) 619 { 620 rect.top += m_MinFirst; 621 rect.bottom -= THICKNESS + m_MinSecond; 622 if (m_DynamicFirst) 623 { 624 if (growth.cy > 0) 625 { 626 m_Pos += min(growth.cy, rect.bottom - (m_Pos + THICKNESS)); 627 } 628 else if (growth.cy < 0) 629 { 630 m_Pos += max(growth.cy, rect.top - m_Pos); 631 } 632 } 633 634 if (m_Pos > rect.bottom) 635 m_Pos = rect.bottom; 636 637 if (m_Pos < rect.top) 638 m_Pos = rect.top; 639 640 splitter.top = m_Pos; 641 splitter.bottom = m_Pos + THICKNESS; 642 first.bottom = splitter.top; 643 second.top = splitter.bottom; 644 } 645 else 646 { 647 rect.left += m_MinFirst; 648 rect.right -= THICKNESS + m_MinSecond; 649 if (m_DynamicFirst) 650 { 651 if (growth.cx > 0) 652 { 653 m_Pos += min(growth.cx, rect.right - (m_Pos + THICKNESS)); 654 } 655 else if (growth.cx < 0) 656 { 657 m_Pos += max(growth.cy, rect.left - m_Pos); 658 } 659 } 660 661 if (m_Pos > rect.right) 662 m_Pos = rect.right; 663 664 if (m_Pos < rect.left) 665 m_Pos = rect.left; 666 667 splitter.left = m_Pos; 668 splitter.right = m_Pos + THICKNESS; 669 first.right = splitter.left; 670 second.left = splitter.right; 671 } 672 673 m_LastRect = parentRect; 674 m_HasOldRect = TRUE; 675 676 hDwp = m_First.OnParentSize(first, hDwp); 677 hDwp = m_Second.OnParentSize(second, hDwp); 678 679 if (hDwp) 680 { 681 return DeferWindowPos(hDwp, NULL, 682 splitter.left, splitter.top, 683 splitter.right - splitter.left, 684 splitter.bottom - splitter.top, 685 SWP_NOACTIVATE | SWP_NOZORDER); 686 } 687 else 688 { 689 SetWindowPos(NULL, 690 splitter.left, splitter.top, 691 splitter.right - splitter.left, 692 splitter.bottom - splitter.top, 693 SWP_NOACTIVATE | SWP_NOZORDER); 694 return NULL; 695 } 696 }; 697 698 private: 699 BOOL ProcessWindowMessage(HWND hwnd, UINT Msg, WPARAM wParam, LPARAM lParam, LRESULT& theResult, DWORD dwMapId) 700 { 701 theResult = 0; 702 switch (Msg) 703 { 704 case WM_SETCURSOR: 705 SetCursor(m_hCursor); 706 theResult = TRUE; 707 break; 708 709 case WM_LBUTTONDOWN: 710 SetCapture(); 711 break; 712 713 case WM_LBUTTONUP: 714 case WM_RBUTTONDOWN: 715 if (GetCapture() == m_hWnd) 716 { 717 ReleaseCapture(); 718 } 719 break; 720 721 case WM_MOUSEMOVE: 722 if (GetCapture() == m_hWnd) 723 { 724 POINT Point; 725 GetCursorPos(&Point); 726 ::ScreenToClient(GetParent(), &Point); 727 if (m_Horizontal) 728 SetPos(Point.y); 729 else 730 SetPos(Point.x); 731 } 732 break; 733 734 default: 735 return FALSE; 736 } 737 738 return TRUE; 739 } 740 741 public: 742 INT GetPos() 743 { 744 return m_Pos; 745 } 746 747 VOID SetPos(INT NewPos) 748 { 749 RECT rcParent; 750 751 rcParent = m_LastRect; 752 753 if (m_Horizontal) 754 { 755 rcParent.bottom -= THICKNESS; 756 757 m_Pos = NewPos; 758 759 if (m_Pos < rcParent.top) 760 m_Pos = rcParent.top; 761 762 if (m_Pos > rcParent.bottom) 763 m_Pos = rcParent.bottom; 764 } 765 else 766 { 767 rcParent.right -= THICKNESS; 768 769 m_Pos = NewPos; 770 771 if (m_Pos < rcParent.left) 772 m_Pos = rcParent.left; 773 774 if (m_Pos > rcParent.right) 775 m_Pos = rcParent.right; 776 } 777 778 INT count = CountSizableChildren(); 779 780 HDWP hdwp = NULL; 781 hdwp = BeginDeferWindowPos(count); 782 if (hdwp) hdwp = OnParentSize(m_LastRect, hdwp); 783 if (hdwp) EndDeferWindowPos(hdwp); 784 } 785 786 public: 787 DECLARE_WND_CLASS_EX(_T("SplitterWindowClass"), CS_HREDRAW | CS_VREDRAW, COLOR_BTNFACE) 788 789 /* Create splitter bar */ 790 HWND Create(HWND hwndParent) 791 { 792 if (m_Horizontal) 793 m_hCursor = LoadCursor(0, IDC_SIZENS); 794 else 795 m_hCursor = LoadCursor(0, IDC_SIZEWE); 796 797 DWORD style = WS_CHILD | WS_VISIBLE; 798 DWORD exStyle = WS_EX_TRANSPARENT; 799 800 RECT size = {205, 180, 465, THICKNESS}; 801 size.right += size.left; 802 size.bottom += size.top; 803 804 return CWindowImpl::Create(hwndParent, size, NULL, style, exStyle); 805 } 806 }; 807