/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* * This file is part of the LibreOffice project. * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ #include #include #include #include #include #include #include #include #include "window.h" #include #include #include #include "svdata.hxx" #include VclContainer::VclContainer(vcl::Window *pParent, WinBits nStyle) : Window(WindowType::CONTAINER) , m_bLayoutDirty(true) { ImplInit(pParent, nStyle, nullptr); EnableChildTransparentMode(); SetPaintTransparent(true); SetBackground(); } sal_uInt16 VclContainer::getDefaultAccessibleRole() const { return css::accessibility::AccessibleRole::PANEL; } Size VclContainer::GetOptimalSize() const { return calculateRequisition(); } void VclContainer::setLayoutPosSize(vcl::Window &rWindow, const Point &rPos, const Size &rSize) { sal_Int32 nBorderWidth = rWindow.get_border_width(); sal_Int32 nLeft = rWindow.get_margin_left() + nBorderWidth; sal_Int32 nTop = rWindow.get_margin_top() + nBorderWidth; sal_Int32 nRight = rWindow.get_margin_right() + nBorderWidth; sal_Int32 nBottom = rWindow.get_margin_bottom() + nBorderWidth; Point aPos(rPos.X() + nLeft, rPos.Y() + nTop); Size aSize(rSize.Width() - nLeft - nRight, rSize.Height() - nTop - nBottom); rWindow.SetPosSizePixel(aPos, aSize); } void VclContainer::setLayoutAllocation(vcl::Window &rChild, const Point &rAllocPos, const Size &rChildAlloc) { VclAlign eHalign = rChild.get_halign(); VclAlign eValign = rChild.get_valign(); //typical case if (eHalign == VclAlign::Fill && eValign == VclAlign::Fill) { setLayoutPosSize(rChild, rAllocPos, rChildAlloc); return; } Point aChildPos(rAllocPos); Size aChildSize(rChildAlloc); Size aChildPreferredSize(getLayoutRequisition(rChild)); switch (eHalign) { case VclAlign::Fill: break; case VclAlign::Start: if (aChildPreferredSize.Width() < rChildAlloc.Width()) aChildSize.Width() = aChildPreferredSize.Width(); break; case VclAlign::End: if (aChildPreferredSize.Width() < rChildAlloc.Width()) aChildSize.Width() = aChildPreferredSize.Width(); aChildPos.X() += rChildAlloc.Width(); aChildPos.X() -= aChildSize.Width(); break; case VclAlign::Center: if (aChildPreferredSize.Width() < aChildSize.Width()) aChildSize.Width() = aChildPreferredSize.Width(); aChildPos.X() += (rChildAlloc.Width() - aChildSize.Width()) / 2; break; } switch (eValign) { case VclAlign::Fill: break; case VclAlign::Start: if (aChildPreferredSize.Height() < rChildAlloc.Height()) aChildSize.Height() = aChildPreferredSize.Height(); break; case VclAlign::End: if (aChildPreferredSize.Height() < rChildAlloc.Height()) aChildSize.Height() = aChildPreferredSize.Height(); aChildPos.Y() += rChildAlloc.Height(); aChildPos.Y() -= aChildSize.Height(); break; case VclAlign::Center: if (aChildPreferredSize.Height() < aChildSize.Height()) aChildSize.Height() = aChildPreferredSize.Height(); aChildPos.Y() += (rChildAlloc.Height() - aChildSize.Height()) / 2; break; } setLayoutPosSize(rChild, aChildPos, aChildSize); } namespace { Size subtractBorder(const vcl::Window &rWindow, const Size& rSize) { sal_Int32 nBorderWidth = rWindow.get_border_width(); sal_Int32 nLeft = rWindow.get_margin_left() + nBorderWidth; sal_Int32 nTop = rWindow.get_margin_top() + nBorderWidth; sal_Int32 nRight = rWindow.get_margin_right() + nBorderWidth; sal_Int32 nBottom = rWindow.get_margin_bottom() + nBorderWidth; Size aSize(rSize); return Size(aSize.Width() + nLeft + nRight, aSize.Height() + nTop + nBottom); } } Size VclContainer::getLayoutRequisition(const vcl::Window &rWindow) { return subtractBorder(rWindow, rWindow.get_preferred_size()); } void VclContainer::SetPosSizePixel(const Point& rAllocPos, const Size& rAllocation) { bool bSizeChanged = rAllocation != GetOutputSizePixel(); Window::SetPosSizePixel(rAllocPos, rAllocation); if (m_bLayoutDirty || bSizeChanged) { m_bLayoutDirty = false; setAllocation(rAllocation); } } void VclContainer::SetPosPixel(const Point& rAllocPos) { Point aAllocPos = rAllocPos; sal_Int32 nBorderWidth = get_border_width(); aAllocPos.X() += nBorderWidth + get_margin_left(); aAllocPos.Y() += nBorderWidth + get_margin_top(); if (aAllocPos != GetPosPixel()) Window::SetPosPixel(aAllocPos); } void VclContainer::SetSizePixel(const Size& rAllocation) { Size aAllocation = rAllocation; sal_Int32 nBorderWidth = get_border_width(); aAllocation.Width() -= nBorderWidth*2 + get_margin_left() + get_margin_right(); aAllocation.Height() -= nBorderWidth*2 + get_margin_top() + get_margin_bottom(); bool bSizeChanged = aAllocation != GetSizePixel(); if (bSizeChanged) Window::SetSizePixel(aAllocation); if (m_bLayoutDirty || bSizeChanged) { m_bLayoutDirty = false; setAllocation(aAllocation); } } void VclContainer::queue_resize(StateChangedType eReason) { m_bLayoutDirty = true; Window::queue_resize(eReason); } Button* isVisibleButtonWithText(vcl::Window* pCandidate) { if (!pCandidate) return nullptr; if (!pCandidate->IsVisible()) return nullptr; if (pCandidate->GetText().isEmpty()) return nullptr; return dynamic_cast(pCandidate); } // evtl. support for screenshot context menu void VclContainer::Command(const CommandEvent& rCEvt) { if (rCEvt.IsMouseEvent() && CommandEventId::ContextMenu == rCEvt.GetCommand()) { const bool bScreenshotMode(officecfg::Office::Common::Misc::ScreenshotMode::get()); if (bScreenshotMode) { bool bVisibleChildren(false); vcl::Window* pChild(nullptr); for (pChild = GetWindow(GetWindowType::FirstChild); !bVisibleChildren && pChild; pChild = pChild->GetWindow(GetWindowType::Next)) { Button* pCandidate = isVisibleButtonWithText(pChild); if (nullptr == pCandidate) continue; bVisibleChildren = true; } if (bVisibleChildren) { static bool bAddButtonsToMenu(true); static bool bAddScreenshotButtonToMenu(true); if (bAddButtonsToMenu || bAddScreenshotButtonToMenu) { const Point aMenuPos(rCEvt.GetMousePosPixel()); ScopedVclPtrInstance aMenu; sal_uInt16 nLocalID(1); sal_uInt16 nScreenshotButtonID(0); if (bAddButtonsToMenu) { for (pChild = GetWindow(GetWindowType::FirstChild); pChild; pChild = pChild->GetWindow(GetWindowType::Next)) { Button* pCandidate = isVisibleButtonWithText(pChild); if (nullptr == pCandidate) continue; aMenu->InsertItem( nLocalID, pChild->GetText()); aMenu->SetHelpText( nLocalID, pChild->GetHelpText()); aMenu->SetHelpId( nLocalID, pChild->GetHelpId()); aMenu->EnableItem( nLocalID, pChild->IsEnabled()); nLocalID++; } } if (bAddScreenshotButtonToMenu) { if (nLocalID > 1) { aMenu->InsertSeparator(); } aMenu->InsertItem( nLocalID, VclResId(SV_BUTTONTEXT_SCREENSHOT)); aMenu->SetHelpText( nLocalID, VclResId(SV_HELPTEXT_SCREENSHOT)); aMenu->SetHelpId( nLocalID, "InteractiveScreenshotMode"); aMenu->EnableItem( nLocalID); nScreenshotButtonID = nLocalID; } const sal_uInt16 nId(aMenu->Execute(this, aMenuPos)); // 0 == no selection (so not usable as ID) if (0 != nId) { if (bAddButtonsToMenu && nId < nLocalID) { nLocalID = 1; for (pChild = GetWindow(GetWindowType::FirstChild); pChild; pChild = pChild->GetWindow(GetWindowType::Next)) { Button* pCandidate = isVisibleButtonWithText(pChild); if (nullptr == pCandidate) continue; if (nLocalID++ == nId) { // pCandidate is the selected button, trigger it pCandidate->Click(); break; } } } if (bAddScreenshotButtonToMenu && nId == nScreenshotButtonID) { // screenshot was selected, access parent dialog (needed for // screenshot and other data access) Dialog* pParentDialog = GetParentDialog(); if (pParentDialog) { // open screenshot annotation dialog VclAbstractDialogFactory* pFact = VclAbstractDialogFactory::Create(); VclPtr pTmp = pFact->CreateScreenshotAnnotationDlg( Application::GetDefDialogParent(), *pParentDialog); ScopedVclPtr pDialog(pTmp); if (pDialog) { // currently just execute the dialog, no need to do // different things for ok/cancel. This may change later, // for that case use 'if (pDlg->Execute() == RET_OK)' pDialog->Execute(); } } } } // consume event when: // - CommandEventId::ContextMenu // - bScreenshotMode // - bVisibleChildren return; } } } } // call parent (do not consume) Window::Command(rCEvt); } void VclBox::accumulateMaxes(const Size &rChildSize, Size &rSize) const { long nSecondaryChildDimension = getSecondaryDimension(rChildSize); long nSecondaryBoxDimension = getSecondaryDimension(rSize); setSecondaryDimension(rSize, std::max(nSecondaryChildDimension, nSecondaryBoxDimension)); long nPrimaryChildDimension = getPrimaryDimension(rChildSize); long nPrimaryBoxDimension = getPrimaryDimension(rSize); if (m_bHomogeneous) setPrimaryDimension(rSize, std::max(nPrimaryBoxDimension, nPrimaryChildDimension)); else setPrimaryDimension(rSize, nPrimaryBoxDimension + nPrimaryChildDimension); } Size VclBox::calculateRequisition() const { sal_uInt16 nVisibleChildren = 0; Size aSize; for (vcl::Window *pChild = GetWindow(GetWindowType::FirstChild); pChild; pChild = pChild->GetWindow(GetWindowType::Next)) { if (!pChild->IsVisible()) continue; ++nVisibleChildren; Size aChildSize = getLayoutRequisition(*pChild); long nPrimaryDimension = getPrimaryDimension(aChildSize); nPrimaryDimension += pChild->get_padding() * 2; setPrimaryDimension(aChildSize, nPrimaryDimension); accumulateMaxes(aChildSize, aSize); } return finalizeMaxes(aSize, nVisibleChildren); } void VclBox::setAllocation(const Size &rAllocation) { sal_uInt16 nVisibleChildren = 0, nExpandChildren = 0; for (vcl::Window *pChild = GetWindow(GetWindowType::FirstChild); pChild; pChild = pChild->GetWindow(GetWindowType::Next)) { if (!pChild->IsVisible()) continue; ++nVisibleChildren; bool bExpand = getPrimaryDimensionChildExpand(*pChild); if (bExpand) ++nExpandChildren; } if (!nVisibleChildren) return; long nAllocPrimaryDimension = getPrimaryDimension(rAllocation); long nHomogeneousDimension = 0, nExtraSpace = 0; if (m_bHomogeneous) { nHomogeneousDimension = ((nAllocPrimaryDimension - (nVisibleChildren - 1) * m_nSpacing)) / nVisibleChildren; } else if (nExpandChildren) { Size aRequisition = calculateRequisition(); nExtraSpace = (getPrimaryDimension(rAllocation) - getPrimaryDimension(aRequisition)) / nExpandChildren; } //Split into those we pack from the start onwards, and those we pack from the end backwards o3tl::enumarray> aWindows; for (vcl::Window *pChild = GetWindow(GetWindowType::FirstChild); pChild; pChild = pChild->GetWindow(GetWindowType::Next)) { if (!pChild->IsVisible()) continue; VclPackType ePacking = pChild->get_pack_type(); aWindows[ePacking].push_back(pChild); } //See VclBuilder::sortIntoBestTabTraversalOrder for why they are in visual //order under the parent which requires us to reverse them here to //pack from the end back std::reverse(aWindows[VclPackType::End].begin(),aWindows[VclPackType::End].end()); for (VclPackType ePackType : o3tl::enumrange()) { Point aPos(0, 0); if (ePackType == VclPackType::End) { long nPrimaryCoordinate = getPrimaryCoordinate(aPos); setPrimaryCoordinate(aPos, nPrimaryCoordinate + nAllocPrimaryDimension); } for (std::vector::iterator aI = aWindows[ePackType].begin(), aEnd = aWindows[ePackType].end(); aI != aEnd; ++aI) { vcl::Window *pChild = *aI; long nPadding = pChild->get_padding(); Size aBoxSize; if (m_bHomogeneous) setPrimaryDimension(aBoxSize, nHomogeneousDimension); else { aBoxSize = getLayoutRequisition(*pChild); long nPrimaryDimension = getPrimaryDimension(aBoxSize); nPrimaryDimension += nPadding * 2; if (getPrimaryDimensionChildExpand(*pChild)) nPrimaryDimension += nExtraSpace; setPrimaryDimension(aBoxSize, nPrimaryDimension); } setSecondaryDimension(aBoxSize, getSecondaryDimension(rAllocation)); Point aChildPos(aPos); Size aChildSize(aBoxSize); long nPrimaryCoordinate = getPrimaryCoordinate(aPos); bool bFill = pChild->get_fill(); if (bFill) { setPrimaryDimension(aChildSize, std::max(static_cast(1), getPrimaryDimension(aBoxSize) - nPadding * 2)); setPrimaryCoordinate(aChildPos, nPrimaryCoordinate + nPadding); } else { setPrimaryDimension(aChildSize, getPrimaryDimension(getLayoutRequisition(*pChild))); setPrimaryCoordinate(aChildPos, nPrimaryCoordinate + (getPrimaryDimension(aBoxSize) - getPrimaryDimension(aChildSize)) / 2); } long nDiff = getPrimaryDimension(aBoxSize) + m_nSpacing; if (ePackType == VclPackType::Start) setPrimaryCoordinate(aPos, nPrimaryCoordinate + nDiff); else { setPrimaryCoordinate(aPos, nPrimaryCoordinate - nDiff); setPrimaryCoordinate(aChildPos, getPrimaryCoordinate(aChildPos) - getPrimaryDimension(aBoxSize)); } setLayoutAllocation(*pChild, aChildPos, aChildSize); } } } bool VclBox::set_property(const OString &rKey, const OUString &rValue) { if (rKey == "spacing") set_spacing(rValue.toInt32()); else if (rKey == "homogeneous") set_homogeneous(toBool(rValue)); else return VclContainer::set_property(rKey, rValue); return true; } sal_uInt16 VclBox::getDefaultAccessibleRole() const { #if defined(_WIN32) //fdo#74284 call Boxes Panels, keep then as "Filler" under //at least Linux seeing as that's what Gtk does for GtkBoxes return css::accessibility::AccessibleRole::PANEL; #else return css::accessibility::AccessibleRole::FILLER; #endif } #define DEFAULT_CHILD_MIN_WIDTH 85 #define DEFAULT_CHILD_MIN_HEIGHT 27 Size VclBox::finalizeMaxes(const Size &rSize, sal_uInt16 nVisibleChildren) const { Size aRet; if (nVisibleChildren) { long nPrimaryDimension = getPrimaryDimension(rSize); if (m_bHomogeneous) nPrimaryDimension *= nVisibleChildren; setPrimaryDimension(aRet, nPrimaryDimension + m_nSpacing * (nVisibleChildren-1)); setSecondaryDimension(aRet, getSecondaryDimension(rSize)); } return aRet; } Size VclButtonBox::addReqGroups(const VclButtonBox::Requisition &rReq) const { Size aRet; long nMainGroupDimension = getPrimaryDimension(rReq.m_aMainGroupSize); long nSubGroupDimension = getPrimaryDimension(rReq.m_aSubGroupSize); setPrimaryDimension(aRet, nMainGroupDimension + nSubGroupDimension); setSecondaryDimension(aRet, std::max(getSecondaryDimension(rReq.m_aMainGroupSize), getSecondaryDimension(rReq.m_aSubGroupSize))); return aRet; } static long getMaxNonOutlier(const std::vector &rG, long nAvgDimension) { long nMaxDimensionNonOutlier = 0; for (std::vector::const_iterator aI = rG.begin(), aEnd = rG.end(); aI != aEnd; ++aI) { long nPrimaryChildDimension = *aI; if (nPrimaryChildDimension < nAvgDimension * 1.5) { nMaxDimensionNonOutlier = std::max(nPrimaryChildDimension, nMaxDimensionNonOutlier); } } return nMaxDimensionNonOutlier; } static std::vector setButtonSizes(const std::vector &rG, const std::vector &rNonHomogeneous, long nAvgDimension, long nMaxNonOutlier, long nMinWidth) { std::vector aVec; //set everything < 1.5 times the average to the same width, leave the //outliers un-touched std::vector::const_iterator aJ = rNonHomogeneous.begin(); for (std::vector::const_iterator aI = rG.begin(), aEnd = rG.end(); aI != aEnd; ++aI, ++aJ) { long nPrimaryChildDimension = *aI; bool bNonHomogeneous = *aJ; if (!bNonHomogeneous && nPrimaryChildDimension < nAvgDimension * 1.5) { aVec.push_back(std::max(nMaxNonOutlier, nMinWidth)); } else { aVec.push_back(std::max(nPrimaryChildDimension, nMinWidth)); } } return aVec; } VclButtonBox::Requisition VclButtonBox::calculatePrimarySecondaryRequisitions() const { Requisition aReq; Size aMainGroupSize(DEFAULT_CHILD_MIN_WIDTH, DEFAULT_CHILD_MIN_HEIGHT); //to-do, pull from theme Size aSubGroupSize(DEFAULT_CHILD_MIN_WIDTH, DEFAULT_CHILD_MIN_HEIGHT); //to-do, pull from theme long nMinMainGroupPrimary = getPrimaryDimension(aMainGroupSize); long nMinSubGroupPrimary = getPrimaryDimension(aSubGroupSize); long nMainGroupSecondary = getSecondaryDimension(aMainGroupSize); long nSubGroupSecondary = getSecondaryDimension(aSubGroupSize); bool bIgnoreSecondaryPacking = (m_eLayoutStyle == VclButtonBoxStyle::Spread || m_eLayoutStyle == VclButtonBoxStyle::Center); std::vector aMainGroupSizes; std::vector aMainGroupNonHomogeneous; std::vector aSubGroupSizes; std::vector aSubGroupNonHomogeneous; for (const vcl::Window *pChild = GetWindow(GetWindowType::FirstChild); pChild; pChild = pChild->GetWindow(GetWindowType::Next)) { if (!pChild->IsVisible()) continue; Size aChildSize = getLayoutRequisition(*pChild); if (bIgnoreSecondaryPacking || !pChild->get_secondary()) { //set the max secondary dimension nMainGroupSecondary = std::max(nMainGroupSecondary, getSecondaryDimension(aChildSize)); //collect the primary dimensions aMainGroupSizes.push_back(getPrimaryDimension(aChildSize)); aMainGroupNonHomogeneous.push_back(pChild->get_non_homogeneous()); } else { nSubGroupSecondary = std::max(nSubGroupSecondary, getSecondaryDimension(aChildSize)); aSubGroupSizes.push_back(getPrimaryDimension(aChildSize)); aSubGroupNonHomogeneous.push_back(pChild->get_non_homogeneous()); } } if (m_bHomogeneous) { long nMaxMainDimension = aMainGroupSizes.empty() ? 0 : *std::max_element(aMainGroupSizes.begin(), aMainGroupSizes.end()); nMaxMainDimension = std::max(nMaxMainDimension, nMinMainGroupPrimary); long nMaxSubDimension = aSubGroupSizes.empty() ? 0 : *std::max_element(aSubGroupSizes.begin(), aSubGroupSizes.end()); nMaxSubDimension = std::max(nMaxSubDimension, nMinSubGroupPrimary); long nMaxDimension = std::max(nMaxMainDimension, nMaxSubDimension); aReq.m_aMainGroupDimensions.resize(aMainGroupSizes.size(), nMaxDimension); aReq.m_aSubGroupDimensions.resize(aSubGroupSizes.size(), nMaxDimension); } else { //Ideally set everything to the same size, but find outlier widgets //that are way wider than the average and leave them //at their natural size and set the remainder to share the //max size of the remaining members of the buttonbox long nAccDimension = std::accumulate(aMainGroupSizes.begin(), aMainGroupSizes.end(), 0); nAccDimension = std::accumulate(aSubGroupSizes.begin(), aSubGroupSizes.end(), nAccDimension); size_t nTotalSize = aMainGroupSizes.size() + aSubGroupSizes.size(); long nAvgDimension = nTotalSize ? nAccDimension / nTotalSize : 0; long nMaxMainNonOutlier = getMaxNonOutlier(aMainGroupSizes, nAvgDimension); long nMaxSubNonOutlier = getMaxNonOutlier(aSubGroupSizes, nAvgDimension); long nMaxNonOutlier = std::max(nMaxMainNonOutlier, nMaxSubNonOutlier); aReq.m_aMainGroupDimensions = setButtonSizes(aMainGroupSizes, aMainGroupNonHomogeneous, nAvgDimension, nMaxNonOutlier, nMinMainGroupPrimary); aReq.m_aSubGroupDimensions = setButtonSizes(aSubGroupSizes, aSubGroupNonHomogeneous, nAvgDimension, nMaxNonOutlier, nMinSubGroupPrimary); } if (!aReq.m_aMainGroupDimensions.empty()) { setSecondaryDimension(aReq.m_aMainGroupSize, nMainGroupSecondary); setPrimaryDimension(aReq.m_aMainGroupSize, std::accumulate(aReq.m_aMainGroupDimensions.begin(), aReq.m_aMainGroupDimensions.end(), 0)); } if (!aReq.m_aSubGroupDimensions.empty()) { setSecondaryDimension(aReq.m_aSubGroupSize, nSubGroupSecondary); setPrimaryDimension(aReq.m_aSubGroupSize, std::accumulate(aReq.m_aSubGroupDimensions.begin(), aReq.m_aSubGroupDimensions.end(), 0)); } return aReq; } Size VclButtonBox::addSpacing(const Size &rSize, sal_uInt16 nVisibleChildren) const { Size aRet; if (nVisibleChildren) { long nPrimaryDimension = getPrimaryDimension(rSize); setPrimaryDimension(aRet, nPrimaryDimension + m_nSpacing * (nVisibleChildren-1)); setSecondaryDimension(aRet, getSecondaryDimension(rSize)); } return aRet; } Size VclButtonBox::calculateRequisition() const { Requisition aReq(calculatePrimarySecondaryRequisitions()); sal_uInt16 nVisibleChildren = aReq.m_aMainGroupDimensions.size() + aReq.m_aSubGroupDimensions.size(); return addSpacing(addReqGroups(aReq), nVisibleChildren); } bool VclButtonBox::set_property(const OString &rKey, const OUString &rValue) { if (rKey == "layout-style") { VclButtonBoxStyle eStyle = VclButtonBoxStyle::Default; if (rValue == "spread") eStyle = VclButtonBoxStyle::Spread; else if (rValue == "edge") eStyle = VclButtonBoxStyle::Edge; else if (rValue == "start") eStyle = VclButtonBoxStyle::Start; else if (rValue == "end") eStyle = VclButtonBoxStyle::End; else if (rValue == "center") eStyle = VclButtonBoxStyle::Center; else { SAL_WARN("vcl.layout", "unknown layout style " << rValue.getStr()); } m_eLayoutStyle = eStyle; } else return VclBox::set_property(rKey, rValue); return true; } void VclButtonBox::setAllocation(const Size &rAllocation) { Requisition aReq(calculatePrimarySecondaryRequisitions()); if (aReq.m_aMainGroupDimensions.empty() && aReq.m_aSubGroupDimensions.empty()) return; long nAllocPrimaryDimension = getPrimaryDimension(rAllocation); Point aMainGroupPos, aOtherGroupPos; int nSpacing = m_nSpacing; //To-Do, other layout styles switch (m_eLayoutStyle) { case VclButtonBoxStyle::Start: if (!aReq.m_aSubGroupDimensions.empty()) { long nOtherPrimaryDimension = getPrimaryDimension( addSpacing(aReq.m_aSubGroupSize, aReq.m_aSubGroupDimensions.size())); setPrimaryCoordinate(aOtherGroupPos, nAllocPrimaryDimension - nOtherPrimaryDimension); } break; case VclButtonBoxStyle::Spread: if (!aReq.m_aMainGroupDimensions.empty()) { long nMainPrimaryDimension = getPrimaryDimension( addSpacing(aReq.m_aMainGroupSize, aReq.m_aMainGroupDimensions.size())); long nExtraSpace = nAllocPrimaryDimension - nMainPrimaryDimension; nExtraSpace += (aReq.m_aMainGroupDimensions.size()-1) * nSpacing; nSpacing = nExtraSpace/(aReq.m_aMainGroupDimensions.size()+1); setPrimaryCoordinate(aMainGroupPos, nSpacing); } break; case VclButtonBoxStyle::Center: if (!aReq.m_aMainGroupDimensions.empty()) { long nMainPrimaryDimension = getPrimaryDimension( addSpacing(aReq.m_aMainGroupSize, aReq.m_aMainGroupDimensions.size())); long nExtraSpace = nAllocPrimaryDimension - nMainPrimaryDimension; setPrimaryCoordinate(aMainGroupPos, nExtraSpace/2); } break; default: SAL_WARN("vcl.layout", "todo unimplemented layout style"); SAL_FALLTHROUGH; case VclButtonBoxStyle::Default: case VclButtonBoxStyle::End: if (!aReq.m_aMainGroupDimensions.empty()) { long nMainPrimaryDimension = getPrimaryDimension( addSpacing(aReq.m_aMainGroupSize, aReq.m_aMainGroupDimensions.size())); setPrimaryCoordinate(aMainGroupPos, nAllocPrimaryDimension - nMainPrimaryDimension); } break; } Size aChildSize; setSecondaryDimension(aChildSize, getSecondaryDimension(rAllocation)); std::vector::const_iterator aPrimaryI = aReq.m_aMainGroupDimensions.begin(); std::vector::const_iterator aSecondaryI = aReq.m_aSubGroupDimensions.begin(); bool bIgnoreSecondaryPacking = (m_eLayoutStyle == VclButtonBoxStyle::Spread || m_eLayoutStyle == VclButtonBoxStyle::Center); for (vcl::Window *pChild = GetWindow(GetWindowType::FirstChild); pChild; pChild = pChild->GetWindow(GetWindowType::Next)) { if (!pChild->IsVisible()) continue; if (bIgnoreSecondaryPacking || !pChild->get_secondary()) { long nMainGroupPrimaryDimension = *aPrimaryI++; setPrimaryDimension(aChildSize, nMainGroupPrimaryDimension); setLayoutAllocation(*pChild, aMainGroupPos, aChildSize); long nPrimaryCoordinate = getPrimaryCoordinate(aMainGroupPos); setPrimaryCoordinate(aMainGroupPos, nPrimaryCoordinate + nMainGroupPrimaryDimension + nSpacing); } else { long nSubGroupPrimaryDimension = *aSecondaryI++; setPrimaryDimension(aChildSize, nSubGroupPrimaryDimension); setLayoutAllocation(*pChild, aOtherGroupPos, aChildSize); long nPrimaryCoordinate = getPrimaryCoordinate(aOtherGroupPos); setPrimaryCoordinate(aOtherGroupPos, nPrimaryCoordinate + nSubGroupPrimaryDimension + nSpacing); } } } struct ButtonOrder { OString m_aType; int m_nPriority; }; static int getButtonPriority(const OString &rType) { static const size_t N_TYPES = 5; static const ButtonOrder aDiscardCancelSave[N_TYPES] = { { "/discard", 0 }, { "/no", 0 }, { "/cancel", 1 }, { "/save", 2 }, { "/yes", 2 } }; static const ButtonOrder aSaveDiscardCancel[N_TYPES] = { { "/save", 0 }, { "/yes", 0 }, { "/discard", 1 }, { "/no", 1 }, { "/cancel", 2 } }; const ButtonOrder* pOrder = &aDiscardCancelSave[0]; const OUString &rEnv = Application::GetDesktopEnvironment(); if (rEnv.equalsIgnoreAsciiCase("windows") || rEnv.equalsIgnoreAsciiCase("tde") || rEnv.startsWithIgnoreAsciiCase("kde")) { pOrder = &aSaveDiscardCancel[0]; } for (size_t i = 0; i < N_TYPES; ++i, ++pOrder) { if (rType.endsWith(pOrder->m_aType)) return pOrder->m_nPriority; } return -1; } class sortButtons : public std::binary_function { bool m_bVerticalContainer; public: explicit sortButtons(bool bVerticalContainer) : m_bVerticalContainer(bVerticalContainer) { } bool operator()(const vcl::Window *pA, const vcl::Window *pB) const; }; bool sortButtons::operator()(const vcl::Window *pA, const vcl::Window *pB) const { //sort into two groups of pack start and pack end VclPackType ePackA = pA->get_pack_type(); VclPackType ePackB = pB->get_pack_type(); if (ePackA < ePackB) return true; if (ePackA > ePackB) return false; bool bPackA = pA->get_secondary(); bool bPackB = pB->get_secondary(); if (!m_bVerticalContainer) { //for horizontal boxes group secondaries before primaries if (bPackA > bPackB) return true; if (bPackA < bPackB) return false; } else { //for vertical boxes group secondaries after primaries if (bPackA < bPackB) return true; if (bPackA > bPackB) return false; } //now order within groups according to platform rules return getButtonPriority(pA->GetHelpId()) < getButtonPriority(pB->GetHelpId()); } void VclButtonBox::sort_native_button_order() { std::vector aChilds; for (vcl::Window* pChild = GetWindow(GetWindowType::FirstChild); pChild; pChild = pChild->GetWindow(GetWindowType::Next)) { aChilds.push_back(pChild); } //sort child order within parent so that we match the platform //button order std::stable_sort(aChilds.begin(), aChilds.end(), sortButtons(m_bVerticalContainer)); VclBuilder::reorderWithinParent(aChilds, true); } struct GridEntry { VclPtr pChild; sal_Int32 nSpanWidth; sal_Int32 nSpanHeight; int x; int y; GridEntry() : pChild(nullptr) , nSpanWidth(0) , nSpanHeight(0) , x(-1) , y(-1) { } }; typedef boost::multi_array array_type; static array_type assembleGrid(const VclGrid &rGrid); static bool isNullGrid(const array_type& A); static void calcMaxs(const array_type &A, std::vector &rWidths, std::vector &rHeights); array_type assembleGrid(const VclGrid &rGrid) { array_type A; for (vcl::Window* pChild = rGrid.GetWindow(GetWindowType::FirstChild); pChild; pChild = pChild->GetWindow(GetWindowType::Next)) { sal_Int32 nLeftAttach = std::max(pChild->get_grid_left_attach(), 0); sal_Int32 nWidth = pChild->get_grid_width(); sal_Int32 nMaxXPos = nLeftAttach+nWidth-1; sal_Int32 nTopAttach = std::max(pChild->get_grid_top_attach(), 0); sal_Int32 nHeight = pChild->get_grid_height(); sal_Int32 nMaxYPos = nTopAttach+nHeight-1; sal_Int32 nCurrentMaxXPos = A.shape()[0]-1; sal_Int32 nCurrentMaxYPos = A.shape()[1]-1; if (nMaxXPos > nCurrentMaxXPos || nMaxYPos > nCurrentMaxYPos) { nCurrentMaxXPos = std::max(nMaxXPos, nCurrentMaxXPos); nCurrentMaxYPos = std::max(nMaxYPos, nCurrentMaxYPos); A.resize(boost::extents[nCurrentMaxXPos+1][nCurrentMaxYPos+1]); } GridEntry &rEntry = A[nLeftAttach][nTopAttach]; rEntry.pChild = pChild; rEntry.nSpanWidth = nWidth; rEntry.nSpanHeight = nHeight; rEntry.x = nLeftAttach; rEntry.y = nTopAttach; for (sal_Int32 nSpanX = 0; nSpanX < nWidth; ++nSpanX) { for (sal_Int32 nSpanY = 0; nSpanY < nHeight; ++nSpanY) { GridEntry &rSpan = A[nLeftAttach+nSpanX][nTopAttach+nSpanY]; rSpan.x = nLeftAttach; rSpan.y = nTopAttach; } } } //see if we have any empty rows/cols sal_Int32 nMaxX = A.shape()[0]; sal_Int32 nMaxY = A.shape()[1]; std::vector aNonEmptyCols(nMaxX); std::vector aNonEmptyRows(nMaxY); for (sal_Int32 x = 0; x < nMaxX; ++x) { for (sal_Int32 y = 0; y < nMaxY; ++y) { const GridEntry &rEntry = A[x][y]; const vcl::Window *pChild = rEntry.pChild; if (pChild && pChild->IsVisible()) { aNonEmptyCols[x] = true; if (rGrid.get_column_homogeneous()) { for (sal_Int32 nSpanX = 1; nSpanX < rEntry.nSpanWidth; ++nSpanX) aNonEmptyCols[x+nSpanX] = true; } aNonEmptyRows[y] = true; if (rGrid.get_row_homogeneous()) { for (sal_Int32 nSpanY = 1; nSpanY < rEntry.nSpanHeight; ++nSpanY) aNonEmptyRows[y+nSpanY] = true; } } } } if (!rGrid.get_column_homogeneous()) { //reduce the spans of elements that span empty columns for (sal_Int32 x = 0; x < nMaxX; ++x) { std::set candidates; for (sal_Int32 y = 0; y < nMaxY; ++y) { if (aNonEmptyCols[x]) continue; GridEntry &rSpan = A[x][y]; //cell x/y is spanned by the widget at cell rSpan.x/rSpan.y, //just points back to itself if there's no cell spanning if ((rSpan.x == -1) || (rSpan.y == -1)) { //there is no entry for this cell, i.e. this is a cell //with no widget in it, or spanned by any other widget continue; } GridEntry &rEntry = A[rSpan.x][rSpan.y]; candidates.insert(&rEntry); } for (std::set::iterator aI = candidates.begin(), aEnd = candidates.end(); aI != aEnd; ++aI) { GridEntry *pEntry = *aI; --pEntry->nSpanWidth; } } } if (!rGrid.get_row_homogeneous()) { //reduce the spans of elements that span empty rows for (sal_Int32 y = 0; y < nMaxY; ++y) { std::set candidates; for (sal_Int32 x = 0; x < nMaxX; ++x) { if (aNonEmptyRows[y]) continue; GridEntry &rSpan = A[x][y]; //cell x/y is spanned by the widget at cell rSpan.x/rSpan.y, //just points back to itself if there's no cell spanning if ((rSpan.x == -1) || (rSpan.y == -1)) { //there is no entry for this cell, i.e. this is a cell //with no widget in it, or spanned by any other widget continue; } GridEntry &rEntry = A[rSpan.x][rSpan.y]; candidates.insert(&rEntry); } for (std::set::iterator aI = candidates.begin(), aEnd = candidates.end(); aI != aEnd; ++aI) { GridEntry *pEntry = *aI; --pEntry->nSpanHeight; } } } sal_Int32 nNonEmptyCols = std::count(aNonEmptyCols.begin(), aNonEmptyCols.end(), true); sal_Int32 nNonEmptyRows = std::count(aNonEmptyRows.begin(), aNonEmptyRows.end(), true); //make new grid without empty rows and columns array_type B(boost::extents[nNonEmptyCols][nNonEmptyRows]); for (sal_Int32 x = 0, x2 = 0; x < nMaxX; ++x) { if (!aNonEmptyCols[x]) continue; for (sal_Int32 y = 0, y2 = 0; y < nMaxY; ++y) { if (!aNonEmptyRows[y]) continue; GridEntry &rEntry = A[x][y]; B[x2][y2++] = rEntry; } ++x2; } return B; } static bool isNullGrid(const array_type &A) { sal_Int32 nMaxX = A.shape()[0]; sal_Int32 nMaxY = A.shape()[1]; return !nMaxX || !nMaxY; } static void calcMaxs(const array_type &A, std::vector &rWidths, std::vector &rHeights) { sal_Int32 nMaxX = A.shape()[0]; sal_Int32 nMaxY = A.shape()[1]; rWidths.resize(nMaxX); rHeights.resize(nMaxY); //first use the non spanning entries to set default width/heights for (sal_Int32 x = 0; x < nMaxX; ++x) { for (sal_Int32 y = 0; y < nMaxY; ++y) { const GridEntry &rEntry = A[x][y]; const vcl::Window *pChild = rEntry.pChild; if (!pChild || !pChild->IsVisible()) continue; sal_Int32 nWidth = rEntry.nSpanWidth; sal_Int32 nHeight = rEntry.nSpanHeight; for (sal_Int32 nSpanX = 0; nSpanX < nWidth; ++nSpanX) rWidths[x+nSpanX].m_bExpand |= pChild->get_hexpand(); for (sal_Int32 nSpanY = 0; nSpanY < nHeight; ++nSpanY) rHeights[y+nSpanY].m_bExpand |= pChild->get_vexpand(); if (nWidth == 1 || nHeight == 1) { Size aChildSize = VclContainer::getLayoutRequisition(*pChild); if (nWidth == 1) rWidths[x].m_nValue = std::max(rWidths[x].m_nValue, aChildSize.Width()); if (nHeight == 1) rHeights[y].m_nValue = std::max(rHeights[y].m_nValue, aChildSize.Height()); } } } //now use the spanning entries and split any extra sizes across expanding rows/cols //where possible for (sal_Int32 x = 0; x < nMaxX; ++x) { for (sal_Int32 y = 0; y < nMaxY; ++y) { const GridEntry &rEntry = A[x][y]; const vcl::Window *pChild = rEntry.pChild; if (!pChild || !pChild->IsVisible()) continue; sal_Int32 nWidth = rEntry.nSpanWidth; sal_Int32 nHeight = rEntry.nSpanHeight; if (nWidth == 1 && nHeight == 1) continue; Size aChildSize = VclContainer::getLayoutRequisition(*pChild); if (nWidth > 1) { sal_Int32 nExistingWidth = 0; for (sal_Int32 nSpanX = 0; nSpanX < nWidth; ++nSpanX) nExistingWidth += rWidths[x+nSpanX].m_nValue; sal_Int32 nExtraWidth = aChildSize.Width() - nExistingWidth; if (nExtraWidth > 0) { bool bForceExpandAll = false; sal_Int32 nExpandables = 0; for (sal_Int32 nSpanX = 0; nSpanX < nWidth; ++nSpanX) if (rWidths[x+nSpanX].m_bExpand) ++nExpandables; if (nExpandables == 0) { nExpandables = nWidth; bForceExpandAll = true; } for (sal_Int32 nSpanX = 0; nSpanX < nWidth; ++nSpanX) { if (rWidths[x+nSpanX].m_bExpand || bForceExpandAll) rWidths[x+nSpanX].m_nValue += nExtraWidth/nExpandables; } } } if (nHeight > 1) { sal_Int32 nExistingHeight = 0; for (sal_Int32 nSpanY = 0; nSpanY < nHeight; ++nSpanY) nExistingHeight += rHeights[y+nSpanY].m_nValue; sal_Int32 nExtraHeight = aChildSize.Height() - nExistingHeight; if (nExtraHeight > 0) { bool bForceExpandAll = false; sal_Int32 nExpandables = 0; for (sal_Int32 nSpanY = 0; nSpanY < nHeight; ++nSpanY) if (rHeights[y+nSpanY].m_bExpand) ++nExpandables; if (nExpandables == 0) { nExpandables = nHeight; bForceExpandAll = true; } for (sal_Int32 nSpanY = 0; nSpanY < nHeight; ++nSpanY) { if (rHeights[y+nSpanY].m_bExpand || bForceExpandAll) rHeights[y+nSpanY].m_nValue += nExtraHeight/nExpandables; } } } } } } bool compareValues(const VclGrid::Value &i, const VclGrid::Value &j) { return i.m_nValue < j.m_nValue; } VclGrid::Value accumulateValues(const VclGrid::Value &i, const VclGrid::Value &j) { VclGrid::Value aRet; aRet.m_nValue = i.m_nValue + j.m_nValue; aRet.m_bExpand = i.m_bExpand || j.m_bExpand; return aRet; } Size VclGrid::calculateRequisition() const { return calculateRequisitionForSpacings(get_row_spacing(), get_column_spacing()); } Size VclGrid::calculateRequisitionForSpacings(sal_Int32 nRowSpacing, sal_Int32 nColSpacing) const { array_type A = assembleGrid(*this); if (isNullGrid(A)) return Size(); std::vector aWidths; std::vector aHeights; calcMaxs(A, aWidths, aHeights); long nTotalWidth = 0; if (get_column_homogeneous()) { nTotalWidth = std::max_element(aWidths.begin(), aWidths.end(), compareValues)->m_nValue; nTotalWidth *= aWidths.size(); } else { nTotalWidth = std::accumulate(aWidths.begin(), aWidths.end(), Value(), accumulateValues).m_nValue; } nTotalWidth += nColSpacing * (aWidths.size()-1); long nTotalHeight = 0; if (get_row_homogeneous()) { nTotalHeight = std::max_element(aHeights.begin(), aHeights.end(), compareValues)->m_nValue; nTotalHeight *= aHeights.size(); } else { nTotalHeight = std::accumulate(aHeights.begin(), aHeights.end(), Value(), accumulateValues).m_nValue; } nTotalHeight += nRowSpacing * (aHeights.size()-1); return Size(nTotalWidth, nTotalHeight); } void VclGrid::setAllocation(const Size& rAllocation) { array_type A = assembleGrid(*this); if (isNullGrid(A)) return; sal_Int32 nMaxX = A.shape()[0]; sal_Int32 nMaxY = A.shape()[1]; Size aRequisition; std::vector aWidths(nMaxX); std::vector aHeights(nMaxY); if (!get_column_homogeneous() || !get_row_homogeneous()) { aRequisition = calculateRequisition(); calcMaxs(A, aWidths, aHeights); } sal_Int32 nColSpacing(get_column_spacing()); sal_Int32 nRowSpacing(get_row_spacing()); long nAvailableWidth = rAllocation.Width(); if (nMaxX) nAvailableWidth -= nColSpacing * (nMaxX - 1); if (get_column_homogeneous()) { for (sal_Int32 x = 0; x < nMaxX; ++x) aWidths[x].m_nValue = nAvailableWidth/nMaxX; } else if (rAllocation.Width() != aRequisition.Width()) { sal_Int32 nExpandables = 0; for (sal_Int32 x = 0; x < nMaxX; ++x) if (aWidths[x].m_bExpand) ++nExpandables; long nExtraWidthForExpanders = nExpandables ? (rAllocation.Width() - aRequisition.Width()) / nExpandables : 0; //We don't fit and there is no volunteer to be shrunk if (!nExpandables && rAllocation.Width() < aRequisition.Width()) { //first reduce spacing while (nColSpacing) { nColSpacing /= 2; aRequisition = calculateRequisitionForSpacings(nRowSpacing, nColSpacing); if (aRequisition.Width() <= rAllocation.Width()) break; } //share out the remaining pain to everyone long nExtraWidth = (rAllocation.Width() - aRequisition.Width()) / nMaxX; for (sal_Int32 x = 0; x < nMaxX; ++x) aWidths[x].m_nValue += nExtraWidth; } if (nExtraWidthForExpanders) { for (sal_Int32 x = 0; x < nMaxX; ++x) if (aWidths[x].m_bExpand) aWidths[x].m_nValue += nExtraWidthForExpanders; } } long nAvailableHeight = rAllocation.Height(); if (nMaxY) nAvailableHeight -= nRowSpacing * (nMaxY - 1); if (get_row_homogeneous()) { for (sal_Int32 y = 0; y < nMaxY; ++y) aHeights[y].m_nValue = nAvailableHeight/nMaxY; } else if (rAllocation.Height() != aRequisition.Height()) { sal_Int32 nExpandables = 0; for (sal_Int32 y = 0; y < nMaxY; ++y) if (aHeights[y].m_bExpand) ++nExpandables; long nExtraHeightForExpanders = nExpandables ? (rAllocation.Height() - aRequisition.Height()) / nExpandables : 0; //We don't fit and there is no volunteer to be shrunk if (!nExpandables && rAllocation.Height() < aRequisition.Height()) { //first reduce spacing while (nRowSpacing) { nRowSpacing /= 2; aRequisition = calculateRequisitionForSpacings(nRowSpacing, nColSpacing); if (aRequisition.Height() <= rAllocation.Height()) break; } //share out the remaining pain to everyone long nExtraHeight = (rAllocation.Height() - aRequisition.Height()) / nMaxY; for (sal_Int32 y = 0; y < nMaxY; ++y) aHeights[y].m_nValue += nExtraHeight; } if (nExtraHeightForExpanders) { for (sal_Int32 y = 0; y < nMaxY; ++y) if (aHeights[y].m_bExpand) aHeights[y].m_nValue += nExtraHeightForExpanders; } } Point aAllocPos(0, 0); for (sal_Int32 x = 0; x < nMaxX; ++x) { for (sal_Int32 y = 0; y < nMaxY; ++y) { GridEntry &rEntry = A[x][y]; vcl::Window *pChild = rEntry.pChild; if (pChild) { Size aChildAlloc(0, 0); sal_Int32 nWidth = rEntry.nSpanWidth; for (sal_Int32 nSpanX = 0; nSpanX < nWidth; ++nSpanX) aChildAlloc.Width() += aWidths[x+nSpanX].m_nValue; aChildAlloc.Width() += nColSpacing*(nWidth-1); sal_Int32 nHeight = rEntry.nSpanHeight; for (sal_Int32 nSpanY = 0; nSpanY < nHeight; ++nSpanY) aChildAlloc.Height() += aHeights[y+nSpanY].m_nValue; aChildAlloc.Height() += nRowSpacing*(nHeight-1); setLayoutAllocation(*pChild, aAllocPos, aChildAlloc); } aAllocPos.Y() += aHeights[y].m_nValue + nRowSpacing; } aAllocPos.X() += aWidths[x].m_nValue + nColSpacing; aAllocPos.Y() = 0; } } bool toBool(const OUString &rValue) { return (!rValue.isEmpty() && (rValue[0] == 't' || rValue[0] == 'T' || rValue[0] == '1')); } bool VclGrid::set_property(const OString &rKey, const OUString &rValue) { if (rKey == "row-spacing") set_row_spacing(rValue.toInt32()); else if (rKey == "column-spacing") set_column_spacing(rValue.toInt32()); else if (rKey == "row-homogeneous") m_bRowHomogeneous = toBool(rValue); else if (rKey == "column-homogeneous") m_bColumnHomogeneous = toBool(rValue); else if (rKey == "n-rows") /*nothing to do*/; else return VclContainer::set_property(rKey, rValue); return true; } const vcl::Window *VclBin::get_child() const { const WindowImpl* pWindowImpl = ImplGetWindowImpl(); return pWindowImpl->mpFirstChild; } vcl::Window *VclBin::get_child() { return const_cast(const_cast(this)->get_child()); } Size VclBin::calculateRequisition() const { const vcl::Window *pChild = get_child(); if (pChild && pChild->IsVisible()) return getLayoutRequisition(*pChild); return Size(0, 0); } void VclBin::setAllocation(const Size &rAllocation) { vcl::Window *pChild = get_child(); if (pChild && pChild->IsVisible()) setLayoutAllocation(*pChild, Point(0, 0), rAllocation); } VclFrame::~VclFrame() { disposeOnce(); } void VclFrame::dispose() { m_pLabel.clear(); VclBin::dispose(); } //To-Do, hook a DecorationView into VclFrame ? Size VclFrame::calculateRequisition() const { Size aRet(0, 0); const vcl::Window *pChild = get_child(); const vcl::Window *pLabel = get_label_widget(); if (pChild && pChild->IsVisible()) aRet = getLayoutRequisition(*pChild); if (pLabel && pLabel->IsVisible()) { Size aLabelSize = getLayoutRequisition(*pLabel); aRet.Height() += aLabelSize.Height(); aRet.Width() = std::max(aLabelSize.Width(), aRet.Width()); } const FrameStyle &rFrameStyle = GetSettings().GetStyleSettings().GetFrameStyle(); aRet.Width() += rFrameStyle.left + rFrameStyle.right; aRet.Height() += rFrameStyle.top + rFrameStyle.bottom; return aRet; } void VclFrame::setAllocation(const Size &rAllocation) { //SetBackground( Color(0xFF, 0x00, 0xFF) ); const FrameStyle &rFrameStyle = GetSettings().GetStyleSettings().GetFrameStyle(); Size aAllocation(rAllocation.Width() - rFrameStyle.left - rFrameStyle.right, rAllocation.Height() - rFrameStyle.top - rFrameStyle.bottom); Point aChildPos(rFrameStyle.left, rFrameStyle.top); vcl::Window *pChild = get_child(); vcl::Window *pLabel = get_label_widget(); if (pLabel && pLabel->IsVisible()) { Size aLabelSize = getLayoutRequisition(*pLabel); aLabelSize.Height() = std::min(aLabelSize.Height(), aAllocation.Height()); aLabelSize.Width() = std::min(aLabelSize.Width(), aAllocation.Width()); setLayoutAllocation(*pLabel, aChildPos, aLabelSize); aAllocation.Height() -= aLabelSize.Height(); aChildPos.Y() += aLabelSize.Height(); } if (pChild && pChild->IsVisible()) setLayoutAllocation(*pChild, aChildPos, aAllocation); } IMPL_LINK(VclFrame, WindowEventListener, VclWindowEvent&, rEvent, void) { if (rEvent.GetId() == VclEventId::ObjectDying) designate_label(nullptr); } void VclFrame::designate_label(vcl::Window *pWindow) { assert(!pWindow || pWindow->GetParent() == this); if (m_pLabel) m_pLabel->RemoveEventListener(LINK(this, VclFrame, WindowEventListener)); m_pLabel = pWindow; if (m_pLabel) m_pLabel->AddEventListener(LINK(this, VclFrame, WindowEventListener)); } const vcl::Window *VclFrame::get_label_widget() const { assert(GetChildCount() == 2); if (m_pLabel) return m_pLabel; //The label widget is normally the first (of two) children const WindowImpl* pWindowImpl = ImplGetWindowImpl(); if (pWindowImpl->mpFirstChild == pWindowImpl->mpLastChild) //no label exists return nullptr; return pWindowImpl->mpFirstChild; } vcl::Window *VclFrame::get_label_widget() { return const_cast(const_cast(this)->get_label_widget()); } const vcl::Window *VclFrame::get_child() const { assert(GetChildCount() == 2); //The child widget is the normally the last (of two) children const WindowImpl* pWindowImpl = ImplGetWindowImpl(); if (!m_pLabel) return pWindowImpl->mpLastChild; if (pWindowImpl->mpFirstChild == pWindowImpl->mpLastChild) //only label exists return nullptr; return pWindowImpl->mpLastChild; } vcl::Window *VclFrame::get_child() { return const_cast(const_cast(this)->get_child()); } void VclFrame::set_label(const OUString &rLabel) { vcl::Window *pLabel = get_label_widget(); assert(pLabel); pLabel->SetText(rLabel); } OUString VclFrame::get_label() const { const vcl::Window *pLabel = get_label_widget(); assert(pLabel); return pLabel->GetText(); } OUString VclFrame::getDefaultAccessibleName() const { const vcl::Window *pLabel = get_label_widget(); if (pLabel) return pLabel->GetAccessibleName(); return VclBin::getDefaultAccessibleName(); } Size VclAlignment::calculateRequisition() const { Size aRet(m_nLeftPadding + m_nRightPadding, m_nTopPadding + m_nBottomPadding); const vcl::Window *pChild = get_child(); if (pChild && pChild->IsVisible()) { Size aChildSize = getLayoutRequisition(*pChild); aRet.Width() += aChildSize.Width(); aRet.Height() += aChildSize.Height(); } return aRet; } void VclAlignment::setAllocation(const Size &rAllocation) { vcl::Window *pChild = get_child(); if (!pChild || !pChild->IsVisible()) return; Point aChildPos(m_nLeftPadding, m_nTopPadding); Size aAllocation; aAllocation.Width() = rAllocation.Width() - (m_nLeftPadding + m_nRightPadding); aAllocation.Height() = rAllocation.Height() - (m_nTopPadding + m_nBottomPadding); setLayoutAllocation(*pChild, aChildPos, aAllocation); } bool VclAlignment::set_property(const OString &rKey, const OUString &rValue) { if (rKey == "bottom-padding") m_nBottomPadding = rValue.toInt32(); else if (rKey == "left-padding") m_nLeftPadding = rValue.toInt32(); else if (rKey == "right-padding") m_nRightPadding = rValue.toInt32(); else if (rKey == "top-padding") m_nTopPadding = rValue.toInt32(); else if (rKey == "xalign") m_fXAlign = rValue.toFloat(); else if (rKey == "xscale") m_fXScale = rValue.toFloat(); else if (rKey == "yalign") m_fYAlign = rValue.toFloat(); else if (rKey == "yscale") m_fYScale = rValue.toFloat(); else return VclBin::set_property(rKey, rValue); return true; } void VclExpander::dispose() { m_pDisclosureButton.disposeAndClear(); VclBin::dispose(); } const vcl::Window *VclExpander::get_child() const { const WindowImpl* pWindowImpl = ImplGetWindowImpl(); assert(pWindowImpl->mpFirstChild == m_pDisclosureButton); return pWindowImpl->mpFirstChild->GetWindow(GetWindowType::Next); } vcl::Window *VclExpander::get_child() { return const_cast(const_cast(this)->get_child()); } Size VclExpander::calculateRequisition() const { Size aRet(0, 0); WindowImpl* pWindowImpl = ImplGetWindowImpl(); const vcl::Window *pChild = get_child(); const vcl::Window *pLabel = pChild != pWindowImpl->mpLastChild ? pWindowImpl->mpLastChild.get() : nullptr; if (pChild && pChild->IsVisible() && m_pDisclosureButton->IsChecked()) aRet = getLayoutRequisition(*pChild); Size aExpanderSize = getLayoutRequisition(*m_pDisclosureButton); if (pLabel && pLabel->IsVisible()) { Size aLabelSize = getLayoutRequisition(*pLabel); aExpanderSize.Height() = std::max(aExpanderSize.Height(), aLabelSize.Height()); aExpanderSize.Width() += aLabelSize.Width(); } aRet.Height() += aExpanderSize.Height(); aRet.Width() = std::max(aExpanderSize.Width(), aRet.Width()); const FrameStyle &rFrameStyle = GetSettings().GetStyleSettings().GetFrameStyle(); aRet.Width() += rFrameStyle.left + rFrameStyle.right; aRet.Height() += rFrameStyle.top + rFrameStyle.bottom; return aRet; } void VclExpander::setAllocation(const Size &rAllocation) { const FrameStyle &rFrameStyle = GetSettings().GetStyleSettings().GetFrameStyle(); Size aAllocation(rAllocation.Width() - rFrameStyle.left - rFrameStyle.right, rAllocation.Height() - rFrameStyle.top - rFrameStyle.bottom); Point aChildPos(rFrameStyle.left, rFrameStyle.top); WindowImpl* pWindowImpl = ImplGetWindowImpl(); //The label widget is the last (of two) children vcl::Window *pChild = get_child(); vcl::Window *pLabel = pChild != pWindowImpl->mpLastChild.get() ? pWindowImpl->mpLastChild.get() : nullptr; Size aButtonSize = getLayoutRequisition(*m_pDisclosureButton); Size aLabelSize; Size aExpanderSize = aButtonSize; if (pLabel && pLabel->IsVisible()) { aLabelSize = getLayoutRequisition(*pLabel); aExpanderSize.Height() = std::max(aExpanderSize.Height(), aLabelSize.Height()); aExpanderSize.Width() += aLabelSize.Width(); } aExpanderSize.Height() = std::min(aExpanderSize.Height(), aAllocation.Height()); aExpanderSize.Width() = std::min(aExpanderSize.Width(), aAllocation.Width()); aButtonSize.Height() = std::min(aButtonSize.Height(), aExpanderSize.Height()); aButtonSize.Width() = std::min(aButtonSize.Width(), aExpanderSize.Width()); long nExtraExpanderHeight = aExpanderSize.Height() - aButtonSize.Height(); Point aButtonPos(aChildPos.X(), aChildPos.Y() + nExtraExpanderHeight/2); setLayoutAllocation(*m_pDisclosureButton, aButtonPos, aButtonSize); if (pLabel && pLabel->IsVisible()) { aLabelSize.Height() = std::min(aLabelSize.Height(), aExpanderSize.Height()); aLabelSize.Width() = std::min(aLabelSize.Width(), aExpanderSize.Width() - aButtonSize.Width()); long nExtraLabelHeight = aExpanderSize.Height() - aLabelSize.Height(); Point aLabelPos(aChildPos.X() + aButtonSize.Width(), aChildPos.Y() + nExtraLabelHeight/2); setLayoutAllocation(*pLabel, aLabelPos, aLabelSize); } aAllocation.Height() -= aExpanderSize.Height(); aChildPos.Y() += aExpanderSize.Height(); if (pChild && pChild->IsVisible()) { if (!m_pDisclosureButton->IsChecked()) aAllocation = Size(); setLayoutAllocation(*pChild, aChildPos, aAllocation); } } bool VclExpander::set_property(const OString &rKey, const OUString &rValue) { if (rKey == "expanded") set_expanded(toBool(rValue)); else if (rKey == "resize-toplevel") m_bResizeTopLevel = toBool(rValue); else return VclBin::set_property(rKey, rValue); return true; } void VclExpander::StateChanged(StateChangedType nType) { VclBin::StateChanged( nType ); if (nType == StateChangedType::InitShow) { vcl::Window *pChild = get_child(); if (pChild) pChild->Show(m_pDisclosureButton->IsChecked()); } } IMPL_LINK( VclExpander, ClickHdl, CheckBox&, rBtn, void ) { vcl::Window *pChild = get_child(); if (pChild) { pChild->Show(rBtn.IsChecked()); queue_resize(); Dialog* pResizeDialog = m_bResizeTopLevel ? GetParentDialog() : nullptr; if (pResizeDialog) pResizeDialog->setOptimalLayoutSize(); } maExpandedHdl.Call(*this); } VclScrolledWindow::VclScrolledWindow(vcl::Window *pParent) : VclBin(pParent, WB_HIDE | WB_CLIPCHILDREN | WB_AUTOHSCROLL | WB_AUTOVSCROLL | WB_TABSTOP) , m_bUserManagedScrolling(false) , m_pVScroll(VclPtr::Create(this, WB_HIDE | WB_VERT)) , m_pHScroll(VclPtr::Create(this, WB_HIDE | WB_HORZ)) , m_aScrollBarBox(VclPtr::Create(this, WB_HIDE)) { SetType(WindowType::SCROLLWINDOW); Link aLink( LINK( this, VclScrolledWindow, ScrollBarHdl ) ); m_pVScroll->SetScrollHdl(aLink); m_pHScroll->SetScrollHdl(aLink); } void VclScrolledWindow::dispose() { m_pVScroll.disposeAndClear(); m_pHScroll.disposeAndClear(); m_aScrollBarBox.disposeAndClear(); VclBin::dispose(); } IMPL_LINK_NOARG(VclScrolledWindow, ScrollBarHdl, ScrollBar*, void) { vcl::Window *pChild = get_child(); if (!pChild) return; assert(dynamic_cast(pChild) && "scrolledwindow child should be a Viewport"); pChild = pChild->GetWindow(GetWindowType::FirstChild); if (!pChild) return; Point aWinPos; if (m_pHScroll->IsVisible()) { aWinPos.X() = -m_pHScroll->GetThumbPos(); } if (m_pVScroll->IsVisible()) { aWinPos.Y() = -m_pVScroll->GetThumbPos(); } pChild->SetPosPixel(aWinPos); } const vcl::Window *VclScrolledWindow::get_child() const { assert(GetChildCount() == 4); const WindowImpl* pWindowImpl = ImplGetWindowImpl(); return pWindowImpl->mpLastChild; } vcl::Window *VclScrolledWindow::get_child() { return const_cast(const_cast(this)->get_child()); } Size VclScrolledWindow::calculateRequisition() const { Size aRet(0, 0); const vcl::Window *pChild = get_child(); if (pChild && pChild->IsVisible()) aRet = getLayoutRequisition(*pChild); if (GetStyle() & WB_VSCROLL) aRet.Width() += getLayoutRequisition(*m_pVScroll).Width(); if (GetStyle() & WB_HSCROLL) aRet.Height() += getLayoutRequisition(*m_pHScroll).Height(); return aRet; } void VclScrolledWindow::InitScrollBars(const Size &rRequest) { const vcl::Window *pChild = get_child(); if (!pChild || !pChild->IsVisible()) return; Size aOutSize(getVisibleChildSize()); if (m_pVScroll->IsVisible()) { m_pVScroll->SetRangeMax(rRequest.Height()); m_pVScroll->SetVisibleSize(aOutSize.Height()); m_pVScroll->SetPageSize(16); } if (m_pHScroll->IsVisible()) { m_pHScroll->SetRangeMax(rRequest.Width()); m_pHScroll->SetVisibleSize(aOutSize.Width()); m_pHScroll->SetPageSize(16); } } void VclScrolledWindow::setAllocation(const Size &rAllocation) { Size aChildAllocation(rAllocation); Size aChildReq; vcl::Window *pChild = get_child(); if (pChild && pChild->IsVisible()) aChildReq = getLayoutRequisition(*pChild); long nAvailHeight = rAllocation.Height(); long nAvailWidth = rAllocation.Width(); // vert. ScrollBar if (GetStyle() & WB_AUTOVSCROLL) { m_pVScroll->Show(nAvailHeight < aChildReq.Height()); } if (m_pVScroll->IsVisible()) nAvailWidth -= getLayoutRequisition(*m_pVScroll).Width(); // horz. ScrollBar if (GetStyle() & WB_AUTOHSCROLL) { bool bShowHScroll = nAvailWidth < aChildReq.Width(); m_pHScroll->Show(bShowHScroll); if (bShowHScroll) nAvailHeight -= getLayoutRequisition(*m_pHScroll).Height(); if (GetStyle() & WB_AUTOVSCROLL) m_pVScroll->Show(nAvailHeight < aChildReq.Height()); } Size aInnerSize(aChildAllocation); long nScrollBarWidth = 0, nScrollBarHeight = 0; if (m_pVScroll->IsVisible()) { nScrollBarWidth = getLayoutRequisition(*m_pVScroll).Width(); Point aScrollPos(rAllocation.Width() - nScrollBarWidth, 0); Size aScrollSize(nScrollBarWidth, rAllocation.Height()); setLayoutAllocation(*m_pVScroll, aScrollPos, aScrollSize); aChildAllocation.Width() -= nScrollBarWidth; aInnerSize.Width() -= nScrollBarWidth; aChildAllocation.Height() = aChildReq.Height(); } if (m_pHScroll->IsVisible()) { nScrollBarHeight = getLayoutRequisition(*m_pHScroll).Height(); Point aScrollPos(0, rAllocation.Height() - nScrollBarHeight); Size aScrollSize(rAllocation.Width(), nScrollBarHeight); setLayoutAllocation(*m_pHScroll, aScrollPos, aScrollSize); aChildAllocation.Height() -= nScrollBarHeight; aInnerSize.Height() -= nScrollBarHeight; aChildAllocation.Width() = aChildReq.Width(); } if (m_pVScroll->IsVisible() && m_pHScroll->IsVisible()) { Point aBoxPos(aInnerSize.Width(), aInnerSize.Height()); m_aScrollBarBox->SetPosSizePixel(aBoxPos, Size(nScrollBarWidth, nScrollBarHeight)); m_aScrollBarBox->Show(); } else { m_aScrollBarBox->Hide(); } if (pChild && pChild->IsVisible()) { assert(dynamic_cast(pChild) && "scrolledwindow child should be a Viewport"); setLayoutAllocation(*pChild, Point(0, 0), aInnerSize); } if (!m_bUserManagedScrolling) InitScrollBars(aChildReq); } Size VclScrolledWindow::getVisibleChildSize() const { Size aRet(GetSizePixel()); if (m_pVScroll->IsVisible()) aRet.Width() -= m_pVScroll->GetSizePixel().Width(); if (m_pHScroll->IsVisible()) aRet.Height() -= m_pHScroll->GetSizePixel().Height(); return aRet; } bool VclScrolledWindow::set_property(const OString &rKey, const OUString &rValue) { bool bRet = VclBin::set_property(rKey, rValue); m_pVScroll->Show((GetStyle() & WB_VSCROLL) != 0); m_pHScroll->Show((GetStyle() & WB_HSCROLL) != 0); return bRet; } bool VclScrolledWindow::EventNotify(NotifyEvent& rNEvt) { bool bDone = false; if ( rNEvt.GetType() == MouseNotifyEvent::COMMAND ) { const CommandEvent& rCEvt = *rNEvt.GetCommandEvent(); if ( rCEvt.GetCommand() == CommandEventId::Wheel ) { const CommandWheelData* pData = rCEvt.GetWheelData(); if( !pData->GetModifier() && ( pData->GetMode() == CommandWheelMode::SCROLL ) ) { bDone = HandleScrollCommand(rCEvt, m_pHScroll, m_pVScroll); } } } return bDone || VclBin::EventNotify( rNEvt ); } void VclViewport::setAllocation(const Size &rAllocation) { vcl::Window *pChild = get_child(); if (pChild && pChild->IsVisible()) { Size aReq(getLayoutRequisition(*pChild)); aReq.Width() = std::max(aReq.Width(), rAllocation.Width()); aReq.Height() = std::max(aReq.Height(), rAllocation.Height()); setLayoutAllocation(*pChild, Point(0, 0), aReq); } } const vcl::Window *VclEventBox::get_child() const { const WindowImpl* pWindowImpl = ImplGetWindowImpl(); assert(pWindowImpl->mpFirstChild.get() == m_aEventBoxHelper.get()); return pWindowImpl->mpFirstChild->GetWindow(GetWindowType::Next); } vcl::Window *VclEventBox::get_child() { return const_cast(const_cast(this)->get_child()); } void VclEventBox::setAllocation(const Size& rAllocation) { Point aChildPos(0, 0); for (vcl::Window *pChild = GetWindow(GetWindowType::FirstChild); pChild; pChild = pChild->GetWindow(GetWindowType::Next)) { if (!pChild->IsVisible()) continue; setLayoutAllocation(*pChild, aChildPos, rAllocation); } } Size VclEventBox::calculateRequisition() const { Size aRet(0, 0); for (const vcl::Window* pChild = get_child(); pChild; pChild = pChild->GetWindow(GetWindowType::Next)) { if (!pChild->IsVisible()) continue; Size aChildSize = getLayoutRequisition(*pChild); aRet.Width() = std::max(aRet.Width(), aChildSize.Width()); aRet.Height() = std::max(aRet.Height(), aChildSize.Height()); } return aRet; } void VclEventBox::Command(const CommandEvent&) { //discard events by default to block them reaching children } VclEventBox::~VclEventBox() { disposeOnce(); } void VclEventBox::dispose() { m_aEventBoxHelper.disposeAndClear(); VclBin::dispose(); } void VclSizeGroup::trigger_queue_resize() { //sufficient to trigger one widget to trigger all of them if (!m_aWindows.empty()) { (*m_aWindows.begin())->queue_resize(); } } void VclSizeGroup::set_ignore_hidden(bool bIgnoreHidden) { if (bIgnoreHidden != m_bIgnoreHidden) { m_bIgnoreHidden = bIgnoreHidden; trigger_queue_resize(); } } void VclSizeGroup::set_mode(VclSizeGroupMode eMode) { if (eMode != m_eMode) { m_eMode = eMode; trigger_queue_resize(); } } bool VclSizeGroup::set_property(const OString &rKey, const OUString &rValue) { if (rKey == "ignore-hidden") set_ignore_hidden(toBool(rValue)); else if (rKey == "mode") { VclSizeGroupMode eMode = VclSizeGroupMode::Horizontal; if (rValue == "none") eMode = VclSizeGroupMode::NONE; else if (rValue == "horizontal") eMode = VclSizeGroupMode::Horizontal; else if (rValue == "vertical") eMode = VclSizeGroupMode::Vertical; else if (rValue == "both") eMode = VclSizeGroupMode::Both; else { SAL_WARN("vcl.layout", "unknown size group mode" << rValue.getStr()); } set_mode(eMode); } else { SAL_INFO("vcl.layout", "unhandled property: " << rKey.getStr()); return false; } return true; } void MessageDialog::create_owned_areas() { set_border_width(12); m_pOwnedContentArea.set(VclPtr::Create(this, false, 24)); set_content_area(m_pOwnedContentArea); m_pOwnedContentArea->Show(); m_pOwnedActionArea.set( VclPtr::Create(m_pOwnedContentArea) ); set_action_area(m_pOwnedActionArea); m_pOwnedActionArea->Show(); } MessageDialog::MessageDialog(vcl::Window* pParent, WinBits nStyle) : Dialog(pParent, nStyle) , m_eButtonsType(VclButtonsType::NONE) , m_eMessageType(VclMessageType::Info) , m_pOwnedContentArea(nullptr) , m_pOwnedActionArea(nullptr) , m_pGrid(nullptr) , m_pImage(nullptr) , m_pPrimaryMessage(nullptr) , m_pSecondaryMessage(nullptr) { SetType(WindowType::MESSBOX); } MessageDialog::MessageDialog(vcl::Window* pParent, const OUString &rMessage, VclMessageType eMessageType, VclButtonsType eButtonsType) : Dialog(pParent, WB_MOVEABLE | WB_3DLOOK | WB_CLOSEABLE) , m_eButtonsType(eButtonsType) , m_eMessageType(eMessageType) , m_pGrid(nullptr) , m_pImage(nullptr) , m_pPrimaryMessage(nullptr) , m_pSecondaryMessage(nullptr) , m_sPrimaryString(rMessage) { SetType(WindowType::MESSBOX); create_owned_areas(); } MessageDialog::MessageDialog(vcl::Window* pParent, const OString& rID, const OUString& rUIXMLDescription) : Dialog(pParent, OStringToOUString(rID, RTL_TEXTENCODING_UTF8), rUIXMLDescription, WindowType::MESSBOX) , m_eButtonsType(VclButtonsType::NONE) , m_eMessageType(VclMessageType::Info) , m_pOwnedContentArea(nullptr) , m_pOwnedActionArea(nullptr) , m_pGrid(nullptr) , m_pImage(nullptr) , m_pPrimaryMessage(nullptr) , m_pSecondaryMessage(nullptr) { } void MessageDialog::dispose() { for (VclPtr & pOwnedButton : m_aOwnedButtons) pOwnedButton.disposeAndClear(); m_aOwnedButtons.clear(); m_pPrimaryMessage.disposeAndClear(); m_pSecondaryMessage.disposeAndClear(); m_pImage.disposeAndClear(); m_pGrid.disposeAndClear(); m_pOwnedActionArea.disposeAndClear(); m_pOwnedContentArea.disposeAndClear(); m_aResponses.clear(); Dialog::dispose(); } MessageDialog::~MessageDialog() { disposeOnce(); } void MessageDialog::response(short nResponseId) { EndDialog(nResponseId); } IMPL_LINK(MessageDialog, ButtonHdl, Button *, pButton, void) { response(get_response(pButton)); } short MessageDialog::get_response(const vcl::Window *pWindow) const { auto aFind = m_aResponses.find(pWindow); if (aFind != m_aResponses.end()) return aFind->second; if (!m_pUIBuilder) return RET_CANCEL; return m_pUIBuilder->get_response(pWindow); } void MessageDialog::setButtonHandlers(VclButtonBox *pButtonBox) { assert(pButtonBox); for (vcl::Window* pChild = pButtonBox->GetWindow(GetWindowType::FirstChild); pChild; pChild = pChild->GetWindow(GetWindowType::Next)) { switch (pChild->GetType()) { case WindowType::PUSHBUTTON: { PushButton* pButton = static_cast(pChild); pButton->SetClickHdl(LINK(this, MessageDialog, ButtonHdl)); break; } //insist that the response ids match the default actions for those //widgets, and leave their default handlers in place case WindowType::OKBUTTON: assert(get_response(pChild) == RET_OK); break; case WindowType::CANCELBUTTON: assert(get_response(pChild) == RET_CANCEL); break; case WindowType::HELPBUTTON: assert(get_response(pChild) == RET_HELP); break; default: SAL_WARN("vcl.layout", "The type of widget " << pChild->GetHelpId() << " is currently not handled"); break; } //The default is to stick the focus into the first widget //that accepts it, and if that happens and it's a button //then that becomes the new default button, so explicitly //put the focus into the default button if (pChild->GetStyle() & WB_DEFBUTTON) pChild->GrabFocus(); } } void MessageDialog::SetMessagesWidths(vcl::Window *pParent, VclMultiLineEdit *pPrimaryMessage, VclMultiLineEdit *pSecondaryMessage) { if (pSecondaryMessage) { assert(pPrimaryMessage); vcl::Font aFont = pParent->GetSettings().GetStyleSettings().GetLabelFont(); aFont.SetFontSize(Size(0, aFont.GetFontSize().Height() * 1.2)); aFont.SetWeight(WEIGHT_BOLD); pPrimaryMessage->SetControlFont(aFont); pPrimaryMessage->SetMaxTextWidth(pPrimaryMessage->approximate_char_width() * 44); pSecondaryMessage->SetMaxTextWidth(pSecondaryMessage->approximate_char_width() * 60); } else pPrimaryMessage->SetMaxTextWidth(pPrimaryMessage->approximate_char_width() * 60); } short MessageDialog::Execute() { setDeferredProperties(); if (!m_pGrid) { VclContainer *pContainer = get_content_area(); assert(pContainer); m_pGrid.set( VclPtr::Create(pContainer) ); m_pGrid->reorderWithinParent(0); m_pGrid->set_column_spacing(12); m_pGrid->set_row_spacing(GetTextHeight()); m_pImage = VclPtr::Create(m_pGrid, WB_CENTER | WB_VCENTER | WB_3DLOOK); switch (m_eMessageType) { case VclMessageType::Info: m_pImage->SetImage(InfoBox::GetStandardImage()); break; case VclMessageType::Warning: m_pImage->SetImage(WarningBox::GetStandardImage()); break; case VclMessageType::Question: m_pImage->SetImage(QueryBox::GetStandardImage()); break; case VclMessageType::Error: m_pImage->SetImage(ErrorBox::GetStandardImage()); break; } m_pImage->set_grid_left_attach(0); m_pImage->set_grid_top_attach(0); m_pImage->set_valign(VclAlign::Start); m_pImage->Show(); WinBits nWinStyle = WB_CLIPCHILDREN | WB_LEFT | WB_VCENTER | WB_NOLABEL | WB_NOTABSTOP; bool bHasSecondaryText = !m_sSecondaryString.isEmpty(); m_pPrimaryMessage = VclPtr::Create(m_pGrid, nWinStyle); m_pPrimaryMessage->SetPaintTransparent(true); m_pPrimaryMessage->EnableCursor(false); m_pPrimaryMessage->set_grid_left_attach(1); m_pPrimaryMessage->set_grid_top_attach(0); m_pPrimaryMessage->set_hexpand(true); m_pPrimaryMessage->SetText(m_sPrimaryString); m_pPrimaryMessage->Show(!m_sPrimaryString.isEmpty()); m_pSecondaryMessage = VclPtr::Create(m_pGrid, nWinStyle); m_pSecondaryMessage->SetPaintTransparent(true); m_pSecondaryMessage->EnableCursor(false); m_pSecondaryMessage->set_grid_left_attach(1); m_pSecondaryMessage->set_grid_top_attach(1); m_pSecondaryMessage->set_hexpand(true); m_pSecondaryMessage->SetText(m_sSecondaryString); m_pSecondaryMessage->Show(bHasSecondaryText); MessageDialog::SetMessagesWidths(this, m_pPrimaryMessage, bHasSecondaryText ? m_pSecondaryMessage.get() : nullptr); VclButtonBox *pButtonBox = get_action_area(); assert(pButtonBox); VclPtr pBtn; switch (m_eButtonsType) { case VclButtonsType::NONE: break; case VclButtonsType::Ok: pBtn.set( VclPtr::Create(pButtonBox) ); pBtn->SetStyle(pBtn->GetStyle() & WB_DEFBUTTON); pBtn->Show(); pBtn->set_id("ok"); m_aOwnedButtons.push_back(pBtn); m_aResponses[pBtn] = RET_OK; break; case VclButtonsType::Close: pBtn.set( VclPtr::Create(pButtonBox) ); pBtn->SetStyle(pBtn->GetStyle() & WB_DEFBUTTON); pBtn->Show(); pBtn->set_id("close"); m_aOwnedButtons.push_back(pBtn); m_aResponses[pBtn] = RET_CLOSE; break; case VclButtonsType::Cancel: pBtn.set( VclPtr::Create(pButtonBox) ); pBtn->SetStyle(pBtn->GetStyle() & WB_DEFBUTTON); pBtn->set_id("cancel"); m_aOwnedButtons.push_back(pBtn); m_aResponses[pBtn] = RET_CANCEL; break; case VclButtonsType::YesNo: pBtn = VclPtr::Create(pButtonBox); pBtn->SetText(Button::GetStandardText(StandardButtonType::Yes)); pBtn->Show(); pBtn->set_id("yes"); m_aOwnedButtons.push_back(pBtn); m_aResponses[pBtn] = RET_YES; pBtn.set( VclPtr::Create(pButtonBox) ); pBtn->SetStyle(pBtn->GetStyle() & WB_DEFBUTTON); pBtn->SetText(Button::GetStandardText(StandardButtonType::No)); pBtn->Show(); pBtn->set_id("no"); m_aOwnedButtons.push_back(pBtn); m_aResponses[pBtn] = RET_NO; break; case VclButtonsType::OkCancel: pBtn.set( VclPtr::Create(pButtonBox) ); pBtn->Show(); pBtn->set_id("ok"); m_aOwnedButtons.push_back(pBtn); m_aResponses[pBtn] = RET_OK; pBtn.set( VclPtr::Create(pButtonBox) ); pBtn->SetStyle(pBtn->GetStyle() & WB_DEFBUTTON); pBtn->Show(); pBtn->set_id("cancel"); m_aOwnedButtons.push_back(pBtn); m_aResponses[pBtn] = RET_CANCEL; break; } setButtonHandlers(pButtonBox); pButtonBox->sort_native_button_order(); m_pGrid->Show(); } return Dialog::Execute(); } OUString const & MessageDialog::get_primary_text() const { const_cast(this)->setDeferredProperties(); return m_sPrimaryString; } OUString const & MessageDialog::get_secondary_text() const { const_cast(this)->setDeferredProperties(); return m_sSecondaryString; } bool MessageDialog::set_property(const OString &rKey, const OUString &rValue) { if (rKey == "text") set_primary_text(rValue); else if (rKey == "secondary-text") set_secondary_text(rValue); else if (rKey == "message-type") { VclMessageType eMode = VclMessageType::Info; if (rValue == "info") eMode = VclMessageType::Info; else if (rValue == "warning") eMode = VclMessageType::Warning; else if (rValue == "question") eMode = VclMessageType::Question; else if (rValue == "error") eMode = VclMessageType::Error; else { SAL_WARN("vcl.layout", "unknown message type mode" << rValue.getStr()); } m_eMessageType = eMode; } else if (rKey == "buttons") { VclButtonsType eMode = VclButtonsType::NONE; if (rValue == "none") eMode = VclButtonsType::NONE; else if (rValue == "ok") eMode = VclButtonsType::Ok; else if (rValue == "cancel") eMode = VclButtonsType::Cancel; else if (rValue == "close") eMode = VclButtonsType::Close; else if (rValue == "yes-no") eMode = VclButtonsType::YesNo; else if (rValue == "ok-cancel") eMode = VclButtonsType::OkCancel; else { SAL_WARN("vcl.layout", "unknown buttons type mode" << rValue.getStr()); } m_eButtonsType = eMode; } else return Dialog::set_property(rKey, rValue); return true; } void MessageDialog::set_primary_text(const OUString &rPrimaryString) { m_sPrimaryString = rPrimaryString; if (m_pPrimaryMessage) { m_pPrimaryMessage->SetText(m_sPrimaryString); m_pPrimaryMessage->Show(!m_sPrimaryString.isEmpty()); } } void MessageDialog::set_secondary_text(const OUString &rSecondaryString) { m_sSecondaryString = rSecondaryString; if (m_pSecondaryMessage) { m_pSecondaryMessage->SetText("\n" + m_sSecondaryString); m_pSecondaryMessage->Show(!m_sSecondaryString.isEmpty()); } } VclVPaned::VclVPaned(vcl::Window *pParent) : VclContainer(pParent, WB_HIDE | WB_CLIPCHILDREN) , m_pSplitter(VclPtr::Create(this, WB_VSCROLL)) , m_nPosition(-1) { m_pSplitter->SetSplitHdl(LINK(this, VclVPaned, SplitHdl)); m_pSplitter->SetBackground(Wallpaper(Application::GetSettings().GetStyleSettings().GetFaceColor())); m_pSplitter->Show(); } void VclVPaned::dispose() { m_pSplitter.disposeAndClear(); VclContainer::dispose(); } IMPL_LINK(VclVPaned, SplitHdl, Splitter*, pSplitter, void) { double nSize = pSplitter->GetSplitPosPixel(); Size aSplitterSize(m_pSplitter->GetSizePixel()); Size aAllocation(GetSizePixel()); arrange(aAllocation, nSize, aAllocation.Height() - nSize - aSplitterSize.Height()); } void VclVPaned::arrange(const Size& rAllocation, long nFirstHeight, long nSecondHeight) { Size aSplitterSize(rAllocation.Width(), getLayoutRequisition(*m_pSplitter).Height()); Size aFirstChildSize(rAllocation.Width(), nFirstHeight); Size aSecondChildSize(rAllocation.Width(), nSecondHeight); int nElement = 0; for (vcl::Window* pChild = GetWindow(GetWindowType::FirstChild); pChild; pChild = pChild->GetWindow(GetWindowType::Next)) { if (!pChild->IsVisible()) continue; if (nElement == 0) { Point aSplitterPos(0, aFirstChildSize.Height()); setLayoutAllocation(*m_pSplitter, aSplitterPos, aSplitterSize); set_position(aSplitterPos.Y() + aSplitterSize.Height() / 2); } else if (nElement == 1) { Point aChildPos(0, 0); setLayoutAllocation(*pChild, aChildPos, aFirstChildSize); } else if (nElement == 2) { Point aChildPos(0, aFirstChildSize.Height() + aSplitterSize.Height()); setLayoutAllocation(*pChild, aChildPos, aSecondChildSize); } ++nElement; } } void VclVPaned::setAllocation(const Size& rAllocation) { //supporting "shrink" could be done by adjusting the allowed drag rectangle m_pSplitter->SetDragRectPixel(tools::Rectangle(Point(0, 0), rAllocation)); Size aSplitterSize(rAllocation.Width(), getLayoutRequisition(*m_pSplitter).Height()); const long nHeight = rAllocation.Height() - aSplitterSize.Height(); long nFirstHeight = 0; long nSecondHeight = 0; bool bFirstCanResize = true; bool bSecondCanResize = true; const bool bInitialAllocation = get_position() < 0; int nElement = 0; for (const vcl::Window* pChild = GetWindow(GetWindowType::FirstChild); pChild; pChild = pChild->GetWindow(GetWindowType::Next)) { if (!pChild->IsVisible()) continue; if (nElement == 1) { if (bInitialAllocation) nFirstHeight = getLayoutRequisition(*pChild).Height(); else nFirstHeight = pChild->GetSizePixel().Height(); bFirstCanResize = pChild->get_expand(); } else if (nElement == 2) { if (bInitialAllocation) nSecondHeight = getLayoutRequisition(*pChild).Height(); else nSecondHeight = pChild->GetSizePixel().Height(); bSecondCanResize = pChild->get_expand(); } ++nElement; } long nHeightRequest = nFirstHeight + nSecondHeight; long nHeightDiff = nHeight - nHeightRequest; if (bFirstCanResize == bSecondCanResize) nFirstHeight += nHeightDiff/2; else if (bFirstCanResize) nFirstHeight += nHeightDiff; arrange(rAllocation, nFirstHeight, nSecondHeight); } Size VclVPaned::calculateRequisition() const { Size aRet(0, 0); for (const vcl::Window* pChild = GetWindow(GetWindowType::FirstChild); pChild; pChild = pChild->GetWindow(GetWindowType::Next)) { if (!pChild->IsVisible()) continue; Size aChildSize = getLayoutRequisition(*pChild); aRet.Width() = std::max(aRet.Width(), aChildSize.Width()); aRet.Height() += aChildSize.Height(); } return aRet; } Size getLegacyBestSizeForChildren(const vcl::Window &rWindow) { tools::Rectangle aBounds; for (const vcl::Window* pChild = rWindow.GetWindow(GetWindowType::FirstChild); pChild; pChild = pChild->GetWindow(GetWindowType::Next)) { if (!pChild->IsVisible()) continue; tools::Rectangle aChildBounds(pChild->GetPosPixel(), pChild->GetSizePixel()); aBounds.Union(aChildBounds); } if (aBounds.IsEmpty()) return rWindow.GetSizePixel(); Size aRet(aBounds.GetSize()); Point aTopLeft(aBounds.TopLeft()); aRet.Width() += aTopLeft.X()*2; aRet.Height() += aTopLeft.Y()*2; return aRet; } vcl::Window* getNonLayoutParent(vcl::Window *pWindow) { while (pWindow) { pWindow = pWindow->GetParent(); if (!pWindow || !isContainerWindow(*pWindow)) break; } return pWindow; } bool isVisibleInLayout(const vcl::Window *pWindow) { bool bVisible = true; while (bVisible) { bVisible = pWindow->IsVisible(); pWindow = pWindow->GetParent(); if (!pWindow || !isContainerWindow(*pWindow)) break; } return bVisible; } bool isEnabledInLayout(const vcl::Window *pWindow) { bool bEnabled = true; while (bEnabled) { bEnabled = pWindow->IsEnabled(); pWindow = pWindow->GetParent(); if (!pWindow || !isContainerWindow(*pWindow)) break; } return bEnabled; } bool isLayoutEnabled(const vcl::Window *pWindow) { //Child is a container => we're layout enabled const vcl::Window *pChild = pWindow ? pWindow->GetWindow(GetWindowType::FirstChild) : nullptr; return pChild && isContainerWindow(*pChild) && !pChild->GetWindow(GetWindowType::Next); } bool isInitialLayout(const vcl::Window *pWindow) { Dialog *pParentDialog = pWindow ? pWindow->GetParentDialog() : nullptr; return pParentDialog && pParentDialog->isCalculatingInitialLayoutSize(); } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */