/* -*- 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/. * * This file incorporates work covered by the following license notice: * * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed * with this work for additional information regarding copyright * ownership. The ASF licenses this file to you under the Apache * License, Version 2.0 (the "License"); you may not use this file * except in compliance with the License. You may obtain a copy of * the License at http://www.apache.org/licenses/LICENSE-2.0 . */ #include "selectlabeldialog.hxx" #include "formresid.hrc" #include "formbrowsertools.hxx" #include "formstrings.hxx" #include #include #include #include #include #include #include #include #include "svtools/treelistentry.hxx" namespace pcr { using namespace ::com::sun::star::uno; using namespace ::com::sun::star::container; using namespace ::com::sun::star::beans; using namespace ::com::sun::star::form; using namespace ::com::sun::star::sdbc; using namespace ::com::sun::star::lang; // OSelectLabelDialog OSelectLabelDialog::OSelectLabelDialog( vcl::Window* pParent, Reference< XPropertySet > _xControlModel ) :ModalDialog(pParent, "LabelSelectionDialog", "modules/spropctrlr/ui/labelselectiondialog.ui") ,m_aModelImages(PcrRes(RID_IL_FORMEXPLORER)) ,m_xControlModel(_xControlModel) ,m_pInitialSelection(NULL) ,m_pLastSelected(NULL) ,m_bHaveAssignableControl(false) { get(m_pMainDesc, "label"); get(m_pControlTree, "control"); get(m_pNoAssignment, "noassignment"); // initialize the TreeListBox m_pControlTree->SetSelectionMode( SINGLE_SELECTION ); m_pControlTree->SetDragDropMode( 0 ); m_pControlTree->EnableInplaceEditing( false ); m_pControlTree->SetStyle(m_pControlTree->GetStyle() | WB_BORDER | WB_HASLINES | WB_HASLINESATROOT | WB_HASBUTTONS | WB_HASBUTTONSATROOT | WB_HSCROLL); m_pControlTree->SetNodeBitmaps( m_aModelImages.GetImage( RID_SVXIMG_COLLAPSEDNODE ), m_aModelImages.GetImage( RID_SVXIMG_EXPANDEDNODE ) ); m_pControlTree->SetSelectHdl(LINK(this, OSelectLabelDialog, OnEntrySelected)); m_pControlTree->SetDeselectHdl(LINK(this, OSelectLabelDialog, OnEntrySelected)); // fill the description OUString sDescription = m_pMainDesc->GetText(); sal_Int16 nClassID = FormComponentType::CONTROL; if (::comphelper::hasProperty(PROPERTY_CLASSID, m_xControlModel)) nClassID = ::comphelper::getINT16(m_xControlModel->getPropertyValue(PROPERTY_CLASSID)); sDescription = sDescription.replaceAll(OUString("$controlclass$"), GetUIHeadlineName(nClassID, makeAny(m_xControlModel))); OUString sName = ::comphelper::getString(m_xControlModel->getPropertyValue(PROPERTY_NAME)); sDescription = sDescription.replaceAll(OUString("$controlname$"), sName); m_pMainDesc->SetText(sDescription); // search for the root of the form hierarchy Reference< XChild > xCont(m_xControlModel, UNO_QUERY); Reference< XInterface > xSearch( xCont.is() ? xCont->getParent() : Reference< XInterface > ()); Reference< XResultSet > xParentAsResultSet(xSearch, UNO_QUERY); while (xParentAsResultSet.is()) { xCont = Reference< XChild > (xSearch, UNO_QUERY); xSearch = xCont.is() ? xCont->getParent() : Reference< XInterface > (); xParentAsResultSet = Reference< XResultSet > (xSearch, UNO_QUERY); } // and insert all entries below this root into the listbox if (xSearch.is()) { // check which service the allowed components must suppport sal_Int16 nClassId = 0; try { nClassId = ::comphelper::getINT16(m_xControlModel->getPropertyValue(PROPERTY_CLASSID)); } catch(...) { } m_sRequiredService = (FormComponentType::RADIOBUTTON == nClassId) ? OUString(SERVICE_COMPONENT_GROUPBOX) : OUString(SERVICE_COMPONENT_FIXEDTEXT); m_aRequiredControlImage = m_aModelImages.GetImage((FormComponentType::RADIOBUTTON == nClassId) ? RID_SVXIMG_GROUPBOX : RID_SVXIMG_FIXEDTEXT); // calc the currently set label control (so InsertEntries can calc m_pInitialSelection) Any aCurrentLabelControl( m_xControlModel->getPropertyValue(PROPERTY_CONTROLLABEL) ); DBG_ASSERT((aCurrentLabelControl.getValueTypeClass() == TypeClass_INTERFACE) || !aCurrentLabelControl.hasValue(), "OSelectLabelDialog::OSelectLabelDialog : invalid ControlLabel property !"); if (aCurrentLabelControl.hasValue()) aCurrentLabelControl >>= m_xInitialLabelControl; // insert the root Image aRootImage = m_aModelImages.GetImage(RID_SVXIMG_FORMS); SvTreeListEntry* pRoot = m_pControlTree->InsertEntry(PcrRes(RID_STR_FORMS).toString(), aRootImage, aRootImage); // build the tree m_pInitialSelection = NULL; m_bHaveAssignableControl = false; InsertEntries(xSearch, pRoot); m_pControlTree->Expand(pRoot); } if (m_pInitialSelection) { m_pControlTree->MakeVisible(m_pInitialSelection, true); m_pControlTree->Select(m_pInitialSelection, true); } else { m_pControlTree->MakeVisible(m_pControlTree->First(), true); if (m_pControlTree->FirstSelected()) m_pControlTree->Select(m_pControlTree->FirstSelected(), false); m_pNoAssignment->Check(true); } if (!m_bHaveAssignableControl) { // no controls which can be assigned m_pNoAssignment->Check(true); m_pNoAssignment->Enable(false); } m_pNoAssignment->SetClickHdl(LINK(this, OSelectLabelDialog, OnNoAssignmentClicked)); m_pNoAssignment->GetClickHdl().Call(m_pNoAssignment); } OSelectLabelDialog::~OSelectLabelDialog() { // delete the entry datas of the listbox entries SvTreeListEntry* pLoop = m_pControlTree->First(); while (pLoop) { void* pData = pLoop->GetUserData(); if (pData) delete static_cast *>(pData); pLoop = m_pControlTree->Next(pLoop); } } sal_Int32 OSelectLabelDialog::InsertEntries(const Reference< XInterface > & _xContainer, SvTreeListEntry* pContainerEntry) { Reference< XIndexAccess > xContainer(_xContainer, UNO_QUERY); if (!xContainer.is()) return 0; sal_Int32 nChildren = 0; OUString sName; Reference< XPropertySet > xAsSet; for (sal_Int32 i=0; igetCount(); ++i) { xContainer->getByIndex(i) >>= xAsSet; if (!xAsSet.is()) { DBG_WARNING("OSelectLabelDialog::InsertEntries : strange : a form component which isn't a property set !"); continue; } if (!::comphelper::hasProperty(PROPERTY_NAME, xAsSet)) // we need at least a name for displaying ... continue; sName = ::comphelper::getString(xAsSet->getPropertyValue(PROPERTY_NAME)).getStr(); // we need to check if the control model supports the required service Reference< XServiceInfo > xInfo(xAsSet, UNO_QUERY); if (!xInfo.is()) continue; if (!xInfo->supportsService(m_sRequiredService)) { // perhaps it is a container Reference< XIndexAccess > xCont(xAsSet, UNO_QUERY); if (xCont.is() && xCont->getCount()) { // yes -> step down Image aFormImage = m_aModelImages.GetImage( RID_SVXIMG_FORM ); SvTreeListEntry* pCont = m_pControlTree->InsertEntry(sName, aFormImage, aFormImage, pContainerEntry); sal_Int32 nContChildren = InsertEntries(xCont, pCont); if (nContChildren) { m_pControlTree->Expand(pCont); ++nChildren; } else { // oops, no valid children -> remove the entry m_pControlTree->ModelIsRemoving(pCont); m_pControlTree->GetModel()->Remove(pCont); m_pControlTree->ModelHasRemoved(pCont); } } continue; } // get the label if (!::comphelper::hasProperty(PROPERTY_LABEL, xAsSet)) continue; OUString sDisplayName = OUStringBuffer( ::comphelper::getString(xAsSet->getPropertyValue(PROPERTY_LABEL))). append(" (").append(sName).append(')'). makeStringAndClear(); // all requirements met -> insert SvTreeListEntry* pCurrent = m_pControlTree->InsertEntry(sDisplayName, m_aRequiredControlImage, m_aRequiredControlImage, pContainerEntry); pCurrent->SetUserData(new Reference< XPropertySet > (xAsSet)); ++nChildren; if (m_xInitialLabelControl == xAsSet) m_pInitialSelection = pCurrent; m_bHaveAssignableControl = true; } return nChildren; } IMPL_LINK(OSelectLabelDialog, OnEntrySelected, SvTreeListBox*, pLB) { DBG_ASSERT(pLB == m_pControlTree, "OSelectLabelDialog::OnEntrySelected : where did this come from ?"); (void)pLB; SvTreeListEntry* pSelected = m_pControlTree->FirstSelected(); void* pData = pSelected ? pSelected->GetUserData() : NULL; if (pData) m_xSelectedControl = Reference< XPropertySet > (*static_cast *>(pData)); m_pNoAssignment->SetClickHdl(Link()); m_pNoAssignment->Check(pData == NULL); m_pNoAssignment->SetClickHdl(LINK(this, OSelectLabelDialog, OnNoAssignmentClicked)); return 0L; } IMPL_LINK(OSelectLabelDialog, OnNoAssignmentClicked, Button*, pButton) { DBG_ASSERT(pButton == m_pNoAssignment, "OSelectLabelDialog::OnNoAssignmentClicked : where did this come from ?"); (void)pButton; if (m_pNoAssignment->IsChecked()) m_pLastSelected = m_pControlTree->FirstSelected(); else { DBG_ASSERT(m_bHaveAssignableControl, "OSelectLabelDialog::OnNoAssignmentClicked"); // search the first assignable entry SvTreeListEntry* pSearch = m_pControlTree->First(); while (pSearch) { if (pSearch->GetUserData()) break; pSearch = m_pControlTree->Next(pSearch); } // and select it if (pSearch) { m_pControlTree->Select(pSearch); m_pLastSelected = pSearch; } } if (m_pLastSelected) { m_pControlTree->SetSelectHdl(Link()); m_pControlTree->SetDeselectHdl(Link()); m_pControlTree->Select(m_pLastSelected, !m_pNoAssignment->IsChecked()); m_pControlTree->SetSelectHdl(LINK(this, OSelectLabelDialog, OnEntrySelected)); m_pControlTree->SetDeselectHdl(LINK(this, OSelectLabelDialog, OnEntrySelected)); } return 0L; } } // namespace pcr /* vim:set shiftwidth=4 softtabstop=4 expandtab: */