/* -*- 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 #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "securitypage.hxx" #include using namespace ::com::sun::star; using namespace ::com::sun::star::uno; struct CustomProperty { OUString m_sName; css::uno::Any m_aValue; CustomProperty( OUString sName, css::uno::Any aValue ) : m_sName(std::move( sName )), m_aValue(std::move( aValue )) {} bool operator==(const CustomProperty& rProp) const { return m_sName == rProp.m_sName && m_aValue == rProp.m_aValue; } }; SfxPoolItem* SfxDocumentInfoItem::CreateDefault() { return new SfxDocumentInfoItem; } namespace { OUString CreateSizeText( sal_Int64 nSize ) { OUString aUnitStr = " " + SfxResId(STR_BYTES); sal_Int64 nSize1 = nSize; sal_Int64 nSize2 = nSize1; sal_Int64 nMega = 1024 * 1024; sal_Int64 nGiga = nMega * 1024; double fSize = nSize; int nDec = 0; if ( nSize1 >= 10000 && nSize1 < nMega ) { nSize1 /= 1024; aUnitStr = " " + SfxResId(STR_KB); fSize /= 1024; nDec = 0; } else if ( nSize1 >= nMega && nSize1 < nGiga ) { nSize1 /= nMega; aUnitStr = " " + SfxResId(STR_MB); fSize /= nMega; nDec = 2; } else if ( nSize1 >= nGiga ) { nSize1 /= nGiga; aUnitStr = " " + SfxResId(STR_GB); fSize /= nGiga; nDec = 3; } const SvtSysLocale aSysLocale; const LocaleDataWrapper& rLocaleWrapper = aSysLocale.GetLocaleData(); OUString aSizeStr = rLocaleWrapper.getNum( nSize1, 0 ) + aUnitStr; if ( nSize1 < nSize2 ) { aSizeStr = ::rtl::math::doubleToUString( fSize, rtl_math_StringFormat_F, nDec, rLocaleWrapper.getNumDecimalSep()[0] ) + aUnitStr + " (" + rLocaleWrapper.getNum( nSize2, 0 ) + " " + SfxResId(STR_BYTES) + ")"; } return aSizeStr; } OUString ConvertDateTime_Impl( std::u16string_view rName, const util::DateTime& uDT, const LocaleDataWrapper& rWrapper ) { Date aD(uDT); tools::Time aT(uDT); static constexpr OUString aDelim( u", "_ustr ); OUString aStr = rWrapper.getDate( aD ) + aDelim + rWrapper.getTime( aT ); std::u16string_view aAuthor = comphelper::string::stripStart(rName, ' '); if (!aAuthor.empty()) { aStr += aDelim + aAuthor; } return aStr; } } SfxDocumentInfoItem::SfxDocumentInfoItem() : m_AutoloadDelay(0) , m_isAutoloadEnabled(false) , m_EditingCycles(0) , m_EditingDuration(0) , m_nFileSize(-1) , m_bHasTemplate( true ) , m_bDeleteUserData( false ) , m_bUseUserData( true ) , m_bUseThumbnailSave( true ) { } SfxDocumentInfoItem::SfxDocumentInfoItem( const OUString& rFile, const uno::Reference& i_xDocProps, const uno::Sequence& i_cmisProps, bool bIs, bool _bIs, sal_Int64 _nFileSize ) : SfxStringItem( SID_DOCINFO, rFile, SfxItemType::SfxDocumentInfoItemType ) , m_AutoloadDelay( i_xDocProps->getAutoloadSecs() ) , m_AutoloadURL( i_xDocProps->getAutoloadURL() ) , m_isAutoloadEnabled( (m_AutoloadDelay > 0) || !m_AutoloadURL.isEmpty() ) , m_DefaultTarget( i_xDocProps->getDefaultTarget() ) , m_TemplateName( i_xDocProps->getTemplateName() ) , m_Author( i_xDocProps->getAuthor() ) , m_CreationDate( i_xDocProps->getCreationDate() ) , m_ModifiedBy( i_xDocProps->getModifiedBy() ) , m_ModificationDate( i_xDocProps->getModificationDate() ) , m_PrintedBy( i_xDocProps->getPrintedBy() ) , m_PrintDate( i_xDocProps->getPrintDate() ) , m_EditingCycles( i_xDocProps->getEditingCycles() ) , m_EditingDuration( i_xDocProps->getEditingDuration() ) , m_Description( i_xDocProps->getDescription() ) , m_Keywords( ::comphelper::string::convertCommaSeparated( i_xDocProps->getKeywords()) ) , m_Subject( i_xDocProps->getSubject() ) , m_Title( i_xDocProps->getTitle() ) , m_nFileSize( _nFileSize ) , m_bHasTemplate( true ) , m_bDeleteUserData( false ) , m_bUseUserData( bIs ) , m_bUseThumbnailSave( _bIs ) { Reference< document::XDocumentProperties2 > xDocProps2(i_xDocProps, UNO_QUERY); m_Contributor = ::comphelper::string::convertCommaSeparated(xDocProps2->getContributor()); m_Coverage = xDocProps2->getCoverage(); m_Identifier = xDocProps2->getIdentifier(); m_Publisher = ::comphelper::string::convertCommaSeparated(xDocProps2->getPublisher()); m_Relation = ::comphelper::string::convertCommaSeparated(xDocProps2->getRelation()); m_Rights = xDocProps2->getRights(); m_Source = xDocProps2->getSource(); m_Type = xDocProps2->getType(); try { Reference< beans::XPropertyContainer > xContainer = i_xDocProps->getUserDefinedProperties(); if ( xContainer.is() ) { Reference < beans::XPropertySet > xSet( xContainer, UNO_QUERY ); const Sequence< beans::Property > lProps = xSet->getPropertySetInfo()->getProperties(); for ( const beans::Property& rProp : lProps ) { // "fix" property? => not a custom property => ignore it! if (!(rProp.Attributes & css::beans::PropertyAttribute::REMOVABLE)) { SAL_WARN( "sfx.dialog", "non-removable user-defined property?"); continue; } uno::Any aValue = xSet->getPropertyValue(rProp.Name); AddCustomProperty( rProp.Name, aValue ); } } // get CMIS properties m_aCmisProperties = i_cmisProps; } catch ( Exception& ) {} } SfxDocumentInfoItem::SfxDocumentInfoItem( const SfxDocumentInfoItem& rItem ) : SfxStringItem( rItem ) , m_AutoloadDelay( rItem.getAutoloadDelay() ) , m_AutoloadURL( rItem.getAutoloadURL() ) , m_isAutoloadEnabled( rItem.isAutoloadEnabled() ) , m_DefaultTarget( rItem.getDefaultTarget() ) , m_TemplateName( rItem.getTemplateName() ) , m_Author( rItem.getAuthor() ) , m_CreationDate( rItem.getCreationDate() ) , m_ModifiedBy( rItem.getModifiedBy() ) , m_ModificationDate( rItem.getModificationDate() ) , m_PrintedBy( rItem.getPrintedBy() ) , m_PrintDate( rItem.getPrintDate() ) , m_EditingCycles( rItem.getEditingCycles() ) , m_EditingDuration( rItem.getEditingDuration() ) , m_Description( rItem.getDescription() ) , m_Keywords( rItem.getKeywords() ) , m_Contributor(rItem.getContributor()) , m_Coverage(rItem.getCoverage()) , m_Identifier(rItem.getIdentifier()) , m_Publisher(rItem.getPublisher()) , m_Relation(rItem.getRelation()) , m_Rights(rItem.getRights()) , m_Source(rItem.getSource()) , m_Type(rItem.getType()) , m_Subject( rItem.getSubject() ) , m_Title( rItem.getTitle() ) , m_nFileSize ( rItem.m_nFileSize ) , m_bHasTemplate( rItem.m_bHasTemplate ) , m_bDeleteUserData( rItem.m_bDeleteUserData ) , m_bUseUserData( rItem.m_bUseUserData ) , m_bUseThumbnailSave( rItem.m_bUseThumbnailSave ) { for (auto const & pOtherProp : rItem.m_aCustomProperties) { AddCustomProperty( pOtherProp->m_sName, pOtherProp->m_aValue ); } m_aCmisProperties = rItem.m_aCmisProperties; } SfxDocumentInfoItem::~SfxDocumentInfoItem() { ClearCustomProperties(); } SfxDocumentInfoItem* SfxDocumentInfoItem::Clone( SfxItemPool * ) const { return new SfxDocumentInfoItem( *this ); } bool SfxDocumentInfoItem::operator==( const SfxPoolItem& rItem) const { if (!SfxStringItem::operator==(rItem)) return false; const SfxDocumentInfoItem& rInfoItem(static_cast(rItem)); return m_AutoloadDelay == rInfoItem.m_AutoloadDelay && m_AutoloadURL == rInfoItem.m_AutoloadURL && m_isAutoloadEnabled == rInfoItem.m_isAutoloadEnabled && m_DefaultTarget == rInfoItem.m_DefaultTarget && m_Author == rInfoItem.m_Author && m_CreationDate == rInfoItem.m_CreationDate && m_ModifiedBy == rInfoItem.m_ModifiedBy && m_ModificationDate == rInfoItem.m_ModificationDate && m_PrintedBy == rInfoItem.m_PrintedBy && m_PrintDate == rInfoItem.m_PrintDate && m_EditingCycles == rInfoItem.m_EditingCycles && m_EditingDuration == rInfoItem.m_EditingDuration && m_Description == rInfoItem.m_Description && m_Keywords == rInfoItem.m_Keywords && m_Contributor == rInfoItem.m_Contributor && m_Coverage == rInfoItem.m_Coverage && m_Identifier == rInfoItem.m_Identifier && m_Publisher == rInfoItem.m_Publisher && m_Relation == rInfoItem.m_Relation && m_Rights == rInfoItem.m_Rights && m_Source == rInfoItem.m_Source && m_Type == rInfoItem.m_Type && m_Subject == rInfoItem.m_Subject && m_Title == rInfoItem.m_Title && comphelper::ContainerUniquePtrEquals(m_aCustomProperties, rInfoItem.m_aCustomProperties) && m_aCmisProperties.getLength() == rInfoItem.m_aCmisProperties.getLength(); } void SfxDocumentInfoItem::resetUserData(const OUString & i_rAuthor) { m_Author = i_rAuthor; DateTime now( DateTime::SYSTEM ); m_CreationDate = now.GetUNODateTime(); m_ModifiedBy = OUString(); m_PrintedBy = OUString(); m_ModificationDate = util::DateTime(); m_PrintDate = util::DateTime(); m_EditingDuration = 0; m_EditingCycles = 1; } void SfxDocumentInfoItem::UpdateDocumentInfo( const uno::Reference& i_xDocProps, bool i_bDoNotUpdateUserDefined) const { if (isAutoloadEnabled()) { i_xDocProps->setAutoloadSecs(getAutoloadDelay()); i_xDocProps->setAutoloadURL(getAutoloadURL()); } else { i_xDocProps->setAutoloadSecs(0); i_xDocProps->setAutoloadURL(OUString()); } i_xDocProps->setDefaultTarget(getDefaultTarget()); i_xDocProps->setAuthor(getAuthor()); i_xDocProps->setCreationDate(getCreationDate()); i_xDocProps->setModifiedBy(getModifiedBy()); i_xDocProps->setModificationDate(getModificationDate()); i_xDocProps->setPrintedBy(getPrintedBy()); i_xDocProps->setPrintDate(getPrintDate()); i_xDocProps->setEditingCycles(getEditingCycles()); i_xDocProps->setEditingDuration(getEditingDuration()); i_xDocProps->setDescription(getDescription()); i_xDocProps->setKeywords( ::comphelper::string::convertCommaSeparated(getKeywords())); Reference xDocProps2(i_xDocProps, UNO_QUERY); xDocProps2->setContributor(::comphelper::string::convertCommaSeparated(getContributor())); xDocProps2->setCoverage(getCoverage()); xDocProps2->setIdentifier(getIdentifier()); xDocProps2->setPublisher(::comphelper::string::convertCommaSeparated(getPublisher())); xDocProps2->setRelation(::comphelper::string::convertCommaSeparated(getRelation())); xDocProps2->setRights(getRights()); xDocProps2->setSource(getSource()); xDocProps2->setType(getType()); xDocProps2->setSubject(getSubject()); xDocProps2->setTitle(getTitle()); // this is necessary in case of replaying a recorded macro: // in this case, the macro may contain the 4 old user-defined DocumentInfo // fields, but not any of the DocumentInfo properties; // as a consequence, most of the UserDefined properties of the // DocumentProperties would be summarily deleted here, which does not // seem like a good idea. if (i_bDoNotUpdateUserDefined) return; try { Reference< beans::XPropertyContainer > xContainer = i_xDocProps->getUserDefinedProperties(); Reference < beans::XPropertySet > xSet( xContainer, UNO_QUERY ); Reference< beans::XPropertySetInfo > xSetInfo = xSet->getPropertySetInfo(); const Sequence< beans::Property > lProps = xSetInfo->getProperties(); for ( const beans::Property& rProp : lProps ) { if (rProp.Attributes & css::beans::PropertyAttribute::REMOVABLE) { xContainer->removeProperty( rProp.Name ); } } for (auto const & pProp : m_aCustomProperties) { try { xContainer->addProperty( pProp->m_sName, beans::PropertyAttribute::REMOVABLE, pProp->m_aValue ); } catch ( Exception const & ) { TOOLS_WARN_EXCEPTION( "sfx.dialog", "SfxDocumentInfoItem::updateDocumentInfo(): exception while adding custom properties" ); } } } catch ( Exception const & ) { TOOLS_WARN_EXCEPTION( "sfx.dialog", "SfxDocumentInfoItem::updateDocumentInfo(): exception while removing custom properties" ); } } void SfxDocumentInfoItem::SetDeleteUserData( bool bSet ) { m_bDeleteUserData = bSet; } void SfxDocumentInfoItem::SetUseUserData( bool bSet ) { m_bUseUserData = bSet; } void SfxDocumentInfoItem::SetUseThumbnailSave( bool bSet ) { m_bUseThumbnailSave = bSet; } std::vector< std::unique_ptr > SfxDocumentInfoItem::GetCustomProperties() const { std::vector< std::unique_ptr > aRet; for (auto const & pOtherProp : m_aCustomProperties) { std::unique_ptr pProp(new CustomProperty( pOtherProp->m_sName, pOtherProp->m_aValue )); aRet.push_back( std::move(pProp) ); } return aRet; } void SfxDocumentInfoItem::ClearCustomProperties() { m_aCustomProperties.clear(); } void SfxDocumentInfoItem::AddCustomProperty( const OUString& sName, const Any& rValue ) { std::unique_ptr pProp(new CustomProperty( sName, rValue )); m_aCustomProperties.push_back( std::move(pProp) ); } void SfxDocumentInfoItem::SetCmisProperties( const Sequence< document::CmisProperty >& cmisProps) { m_aCmisProperties = cmisProps; } bool SfxDocumentInfoItem::QueryValue( Any& rVal, sal_uInt8 nMemberId ) const { OUString aValue; sal_Int32 nValue = 0; bool bValue = false; bool bIsInt = false; bool bIsString = false; nMemberId &= ~CONVERT_TWIPS; switch ( nMemberId ) { case MID_DOCINFO_USEUSERDATA: bValue = IsUseUserData(); break; case MID_DOCINFO_USETHUMBNAILSAVE: bValue = IsUseThumbnailSave(); break; case MID_DOCINFO_DELETEUSERDATA: bValue = m_bDeleteUserData; break; case MID_DOCINFO_AUTOLOADENABLED: bValue = isAutoloadEnabled(); break; case MID_DOCINFO_AUTOLOADSECS: bIsInt = true; nValue = getAutoloadDelay(); break; case MID_DOCINFO_AUTOLOADURL: bIsString = true; aValue = getAutoloadURL(); break; case MID_DOCINFO_DEFAULTTARGET: bIsString = true; aValue = getDefaultTarget(); break; case MID_DOCINFO_DESCRIPTION: bIsString = true; aValue = getDescription(); break; case MID_DOCINFO_KEYWORDS: bIsString = true; aValue = getKeywords(); break; case MID_DOCINFO_SUBJECT: bIsString = true; aValue = getSubject(); break; case MID_DOCINFO_TITLE: bIsString = true; aValue = getTitle(); break; default: OSL_FAIL("Wrong MemberId!"); return false; } if ( bIsString ) rVal <<= aValue; else if ( bIsInt ) rVal <<= nValue; else rVal <<= bValue; return true; } bool SfxDocumentInfoItem::PutValue( const Any& rVal, sal_uInt8 nMemberId ) { OUString aValue; sal_Int32 nValue=0; bool bValue = false; bool bRet = false; nMemberId &= ~CONVERT_TWIPS; switch ( nMemberId ) { case MID_DOCINFO_USEUSERDATA: bRet = (rVal >>= bValue); if ( bRet ) SetUseUserData( bValue ); break; case MID_DOCINFO_USETHUMBNAILSAVE: bRet = (rVal >>=bValue); if ( bRet ) SetUseThumbnailSave( bValue ); break; case MID_DOCINFO_DELETEUSERDATA: // QUESTION: deleting user data was done here; seems to be superfluous! bRet = (rVal >>= bValue); if ( bRet ) SetDeleteUserData( bValue ); break; case MID_DOCINFO_AUTOLOADENABLED: bRet = (rVal >>= bValue); if ( bRet ) m_isAutoloadEnabled = bValue; break; case MID_DOCINFO_AUTOLOADSECS: bRet = (rVal >>= nValue); if ( bRet ) m_AutoloadDelay = nValue; break; case MID_DOCINFO_AUTOLOADURL: bRet = (rVal >>= aValue); if ( bRet ) m_AutoloadURL = aValue; break; case MID_DOCINFO_DEFAULTTARGET: bRet = (rVal >>= aValue); if ( bRet ) m_DefaultTarget = aValue; break; case MID_DOCINFO_DESCRIPTION: bRet = (rVal >>= aValue); if ( bRet ) setDescription(aValue); break; case MID_DOCINFO_KEYWORDS: bRet = (rVal >>= aValue); if ( bRet ) setKeywords(aValue); break; case MID_DOCINFO_SUBJECT: bRet = (rVal >>= aValue); if ( bRet ) setSubject(aValue); break; case MID_DOCINFO_TITLE: bRet = (rVal >>= aValue); if ( bRet ) setTitle(aValue); break; default: OSL_FAIL("Wrong MemberId!"); return false; } return bRet; } SfxDocumentDescPage::SfxDocumentDescPage(weld::Container* pPage, weld::DialogController* pController, const SfxItemSet& rItemSet) : SfxTabPage(pPage, pController, u"sfx/ui/descriptioninfopage.ui"_ustr, u"DescriptionInfoPage"_ustr, &rItemSet) , m_pInfoItem(nullptr) , m_xTitleEd(m_xBuilder->weld_entry(u"title"_ustr)) , m_xThemaEd(m_xBuilder->weld_entry(u"subject"_ustr)) , m_xKeywordsEd(m_xBuilder->weld_entry(u"keywords"_ustr)) , m_xContributorEd(m_xBuilder->weld_entry(u"contributor"_ustr)) , m_xCoverageEd(m_xBuilder->weld_entry(u"coverage"_ustr)) , m_xIdentifierEd(m_xBuilder->weld_entry(u"identifier"_ustr)) , m_xPublisherEd(m_xBuilder->weld_entry(u"publisher"_ustr)) , m_xRelationEd(m_xBuilder->weld_entry(u"relation"_ustr)) , m_xRightsEd(m_xBuilder->weld_entry(u"rights"_ustr)) , m_xSourceEd(m_xBuilder->weld_entry(u"source"_ustr)) , m_xTypeEd(m_xBuilder->weld_entry(u"type"_ustr)) , m_xCommentEd(m_xBuilder->weld_text_view(u"comments"_ustr)) { m_xCommentEd->set_size_request(m_xKeywordsEd->get_preferred_size().Width(), m_xCommentEd->get_height_rows(16)); } SfxDocumentDescPage::~SfxDocumentDescPage() { } std::unique_ptr SfxDocumentDescPage::Create(weld::Container* pPage, weld::DialogController* pController, const SfxItemSet *rItemSet) { return std::make_unique(pPage, pController, *rItemSet); } bool SfxDocumentDescPage::FillItemSet(SfxItemSet *rSet) { // Test whether a change is present const bool bTitleMod = m_xTitleEd->get_value_changed_from_saved(); const bool bThemeMod = m_xThemaEd->get_value_changed_from_saved(); const bool bKeywordsMod = m_xKeywordsEd->get_value_changed_from_saved(); const bool bContributorMod = m_xContributorEd->get_value_changed_from_saved(); const bool bCoverageMod = m_xCoverageEd->get_value_changed_from_saved(); const bool bIdentifierMod = m_xIdentifierEd->get_value_changed_from_saved(); const bool bPublisherMod = m_xPublisherEd->get_value_changed_from_saved(); const bool bRelationMod = m_xRelationEd->get_value_changed_from_saved(); const bool bRightsMod = m_xRightsEd->get_value_changed_from_saved(); const bool bSourceMod = m_xSourceEd->get_value_changed_from_saved(); const bool bTypeMod = m_xTypeEd->get_value_changed_from_saved(); const bool bCommentMod = m_xCommentEd->get_value_changed_from_saved(); if (!(bTitleMod || bThemeMod || bKeywordsMod || bContributorMod || bCoverageMod || bIdentifierMod || bPublisherMod || bRelationMod || bRightsMod || bSourceMod || bTypeMod || bCommentMod)) { return false; } // Generating the output data const SfxDocumentInfoItem* pItem = nullptr; SfxDocumentInfoItem* pInfo = nullptr; const SfxItemSet* pExSet = GetDialogExampleSet(); if ( pExSet && !(pItem = pExSet->GetItemIfSet( SID_DOCINFO )) ) pInfo = m_pInfoItem; else if ( pItem ) pInfo = new SfxDocumentInfoItem( *pItem ); if ( !pInfo ) { SAL_WARN( "sfx.dialog", "SfxDocumentDescPage::FillItemSet(): no item found" ); return false; } if ( bTitleMod ) { pInfo->setTitle( m_xTitleEd->get_text() ); } if ( bThemeMod ) { pInfo->setSubject( m_xThemaEd->get_text() ); } if ( bKeywordsMod ) { pInfo->setKeywords( m_xKeywordsEd->get_text() ); } if (bContributorMod) { pInfo->setContributor(m_xContributorEd->get_text()); } if (bCoverageMod) { pInfo->setCoverage(m_xCoverageEd->get_text()); } if (bIdentifierMod) { pInfo->setIdentifier(m_xIdentifierEd->get_text()); } if (bPublisherMod) { pInfo->setPublisher(m_xPublisherEd->get_text()); } if (bRelationMod) { pInfo->setRelation(m_xRelationEd->get_text()); } if (bRightsMod) { pInfo->setRights(m_xRightsEd->get_text()); } if (bSourceMod) { pInfo->setSource(m_xSourceEd->get_text()); } if (bTypeMod) { pInfo->setType(m_xTypeEd->get_text()); } if ( bCommentMod ) { pInfo->setDescription( m_xCommentEd->get_text() ); } rSet->Put( *pInfo ); // ptr compare OK, pInfo was created above as temporary data holder if ( !areSfxPoolItemPtrsEqual(pInfo, m_pInfoItem) ) { delete pInfo; } return true; } void SfxDocumentDescPage::Reset(const SfxItemSet *rSet) { m_pInfoItem = const_cast(&rSet->Get(SID_DOCINFO)); m_xTitleEd->set_text(m_pInfoItem->getTitle()); m_xThemaEd->set_text(m_pInfoItem->getSubject()); m_xKeywordsEd->set_text(m_pInfoItem->getKeywords()); m_xContributorEd->set_text(m_pInfoItem->getContributor()); m_xCoverageEd->set_text(m_pInfoItem->getCoverage()); m_xIdentifierEd->set_text(m_pInfoItem->getIdentifier()); m_xPublisherEd->set_text(m_pInfoItem->getPublisher()); m_xRelationEd->set_text(m_pInfoItem->getRelation()); m_xRightsEd->set_text(m_pInfoItem->getRights()); m_xSourceEd->set_text(m_pInfoItem->getSource()); m_xTypeEd->set_text(m_pInfoItem->getType()); m_xCommentEd->set_text(m_pInfoItem->getDescription()); m_xTitleEd->save_value(); m_xThemaEd->save_value(); m_xKeywordsEd->save_value(); m_xContributorEd->save_value(); m_xCoverageEd->save_value(); m_xIdentifierEd->save_value(); m_xPublisherEd->save_value(); m_xRelationEd->save_value(); m_xRightsEd->save_value(); m_xSourceEd->save_value(); m_xTypeEd->save_value(); m_xCommentEd->save_value(); const SfxBoolItem* pROItem = SfxItemSet::GetItem(rSet, SID_DOC_READONLY, false); if (pROItem && pROItem->GetValue()) { m_xTitleEd->set_editable(false); m_xThemaEd->set_editable(false); m_xKeywordsEd->set_editable(false); m_xContributorEd->set_editable(false); m_xCoverageEd->set_editable(false); m_xIdentifierEd->set_editable(false); m_xPublisherEd->set_editable(false); m_xRelationEd->set_editable(false); m_xRightsEd->set_editable(false); m_xSourceEd->set_editable(false); m_xTypeEd->set_editable(false); m_xCommentEd->set_editable(false); } } SfxDocumentPage::SfxDocumentPage(weld::Container* pPage, weld::DialogController* pController, const SfxItemSet& rItemSet) : SfxTabPage(pPage, pController, u"sfx/ui/documentinfopage.ui"_ustr, u"DocumentInfoPage"_ustr, &rItemSet) , bEnableUseUserData( false ) , bHandleDelete( false ) , m_xBmp(m_xBuilder->weld_image(u"icon"_ustr)) , m_xNameED(m_xBuilder->weld_label(u"nameed"_ustr)) , m_xChangePassBtn(m_xBuilder->weld_button(u"changepass"_ustr)) , m_xShowTypeFT(m_xBuilder->weld_label(u"showtype"_ustr)) , m_xFileValEd(m_xBuilder->weld_link_button(u"showlocation"_ustr)) , m_xShowSizeFT(m_xBuilder->weld_label(u"showsize"_ustr)) , m_xCreateValFt(m_xBuilder->weld_label(u"showcreate"_ustr)) , m_xChangeValFt(m_xBuilder->weld_label(u"showmodify"_ustr)) , m_xSignedValFt(m_xBuilder->weld_label(u"showsigned"_ustr)) , m_xSignatureBtn(m_xBuilder->weld_button(u"signature"_ustr)) , m_xPrintValFt(m_xBuilder->weld_label(u"showprint"_ustr)) , m_xTimeLogValFt(m_xBuilder->weld_label(u"showedittime"_ustr)) , m_xDocNoValFt(m_xBuilder->weld_label(u"showrevision"_ustr)) , m_xUseUserDataCB(m_xBuilder->weld_check_button(u"userdatacb"_ustr)) , m_xDeleteBtn(m_xBuilder->weld_button(u"reset"_ustr)) , m_xUseThumbnailSaveCB(m_xBuilder->weld_check_button(u"thumbnailsavecb"_ustr)) , m_xTemplFt(m_xBuilder->weld_label(u"templateft"_ustr)) , m_xTemplValFt(m_xBuilder->weld_label(u"showtemplate"_ustr)) , m_xImagePreferredDpiCheckButton(m_xBuilder->weld_check_button(u"image-preferred-dpi-checkbutton"_ustr)) , m_xImagePreferredDpiComboBox(m_xBuilder->weld_combo_box(u"image-preferred-dpi-combobox"_ustr)) { m_xUseUserDataCB->set_accessible_description(SfxResId(STR_A11Y_DESC_USERDATA)); m_aUnknownSize = m_xShowSizeFT->get_label(); m_xShowSizeFT->set_label(OUString()); m_aMultiSignedStr = m_xSignedValFt->get_label(); m_xSignedValFt->set_label(OUString()); m_xImagePreferredDpiComboBox->set_entry_width_chars(5); // "width in chars" in Glade not effective ImplUpdateSignatures(); ImplCheckPasswordState(); m_xChangePassBtn->connect_clicked( LINK( this, SfxDocumentPage, ChangePassHdl ) ); m_xSignatureBtn->connect_clicked( LINK( this, SfxDocumentPage, SignatureHdl ) ); if (comphelper::LibreOfficeKit::isActive()) m_xSignatureBtn->hide(); m_xDeleteBtn->connect_clicked( LINK( this, SfxDocumentPage, DeleteHdl ) ); m_xImagePreferredDpiCheckButton->connect_toggled(LINK(this, SfxDocumentPage, ImagePreferredDPICheckBoxClicked)); // [i96288] Check if the document signature command is enabled // on the main list enable/disable the pushbutton accordingly SvtCommandOptions aCmdOptions; if ( aCmdOptions.LookupDisabled( u"Signature"_ustr ) ) m_xSignatureBtn->set_sensitive(false); } SfxDocumentPage::~SfxDocumentPage() { if (m_xPasswordDialog) { m_xPasswordDialog->Response(RET_CANCEL); m_xPasswordDialog.clear(); } } IMPL_LINK_NOARG(SfxDocumentPage, DeleteHdl, weld::Button&, void) { OUString aName; if (bEnableUseUserData && m_xUseUserDataCB->get_active()) aName = SvtUserOptions().GetFullName(); const LocaleDataWrapper& rLocaleWrapper( Application::GetSettings().GetLocaleDataWrapper() ); DateTime now( DateTime::SYSTEM ); util::DateTime uDT( now.GetUNODateTime() ); m_xCreateValFt->set_label( ConvertDateTime_Impl( aName, uDT, rLocaleWrapper ) ); m_xChangeValFt->set_label( u""_ustr ); m_xPrintValFt->set_label( u""_ustr ); m_xTimeLogValFt->set_label( rLocaleWrapper.getDuration( tools::Duration() ) ); m_xDocNoValFt->set_label(OUString('1')); bHandleDelete = true; } IMPL_LINK_NOARG(SfxDocumentPage, SignatureHdl, weld::Button&, void) { SfxObjectShell* pDoc = SfxObjectShell::Current(); if( pDoc ) { pDoc->SignDocumentContent(GetFrameWeld(), [this] (bool /*bHaveWeSigned*/) { ImplUpdateSignatures(); }); } } IMPL_LINK_NOARG(SfxDocumentPage, ImagePreferredDPICheckBoxClicked, weld::Toggleable&, void) { bool bEnabled = m_xImagePreferredDpiCheckButton->get_state() == TRISTATE_TRUE; m_xImagePreferredDpiComboBox->set_sensitive(bEnabled); } IMPL_LINK_NOARG(SfxDocumentPage, ChangePassHdl, weld::Button&, void) { SfxObjectShell* pShell = SfxObjectShell::Current(); do { if (!pShell) break; SfxItemSet& rMedSet = pShell->GetMedium()->GetItemSet(); std::shared_ptr pFilter = pShell->GetMedium()->GetFilter(); if (!pFilter) break; if (comphelper::LibreOfficeKit::isActive()) { // MS Types support max len of 15 characters while OOXML is "unlimited" const sal_uInt16 maxPwdLen = sfx2::IsMSType(pFilter) && !sfx2::IsOOXML(pFilter) ? 15 : 0; // handle the pwd dialog asynchronously VclAbstractDialogFactory * pFact = VclAbstractDialogFactory::Create(); m_xPasswordDialog = pFact->CreatePasswordToOpenModifyDialog(GetFrameWeld(), maxPwdLen, false); m_xPasswordDialog->AllowEmpty(); // needed to remove password m_xPasswordDialog->StartExecuteAsync([this, pFilter=std::move(pFilter), &rMedSet, pShell](sal_Int32 nResult) { if (nResult == RET_OK) { sfx2::SetPassword(pFilter, &rMedSet, m_xPasswordDialog->GetPasswordToOpen(), m_xPasswordDialog->GetPasswordToOpen(), true); tools::JsonWriter payloadJson; payloadJson.put("password", m_xPasswordDialog->GetPasswordToOpen()); payloadJson.put("isToModify", false); SfxViewShell *pViewShell = SfxViewShell::Current(); if (pViewShell) pViewShell->libreOfficeKitViewCallback(LOK_CALLBACK_DOCUMENT_PASSWORD_RESET, payloadJson.finishAndGetAsOString()); pShell->SetModified(); } m_xPasswordDialog->disposeOnce(); }); } else { sfx2::RequestPassword(pFilter, OUString(), &rMedSet, GetFrameWeld()->GetXWindow()); pShell->SetModified(); } } while (false); } void SfxDocumentPage::ImplUpdateSignatures() { SfxObjectShell* pDoc = SfxObjectShell::Current(); if ( !pDoc ) return; SfxMedium* pMedium = pDoc->GetMedium(); if ( !pMedium || pMedium->GetName().isEmpty() || !pMedium->GetStorage().is() ) return; Reference< security::XDocumentDigitalSignatures > xD; try { xD = security::DocumentDigitalSignatures::createDefault(comphelper::getProcessComponentContext()); xD->setParentWindow(GetDialogController()->getDialog()->GetXWindow()); } catch ( const css::uno::DeploymentException& ) { } OUString s; Sequence< security::DocumentSignatureInformation > aInfos; if ( xD.is() ) aInfos = xD->verifyDocumentContentSignatures( pMedium->GetZipStorageToSign_Impl(), uno::Reference< io::XInputStream >() ); if ( aInfos.getLength() > 1 ) s = m_aMultiSignedStr; else if ( aInfos.getLength() == 1 ) { const security::DocumentSignatureInformation& rInfo = aInfos[ 0 ]; s = utl::GetDateTimeString( rInfo.SignatureDate, rInfo.SignatureTime ) + ", " + comphelper::xmlsec::GetContentPart(rInfo.Signer->getSubjectName(), rInfo.Signer->getCertificateKind()); } m_xSignedValFt->set_label(s); } void SfxDocumentPage::ImplCheckPasswordState() { SfxObjectShell* pShell = SfxObjectShell::Current(); do { if (!pShell) break; const SfxUnoAnyItem* pEncryptionDataItem = pShell->GetMedium()->GetItemSet().GetItem(SID_ENCRYPTIONDATA, false); uno::Sequence< beans::NamedValue > aEncryptionData; if (pEncryptionDataItem) pEncryptionDataItem->GetValue() >>= aEncryptionData; else break; if (!aEncryptionData.hasElements()) break; m_xChangePassBtn->set_sensitive(true); return; } while (false); m_xChangePassBtn->set_sensitive(comphelper::LibreOfficeKit::isActive()); } std::unique_ptr SfxDocumentPage::Create(weld::Container* pPage, weld::DialogController* pController, const SfxItemSet* rItemSet) { return std::make_unique(pPage, pController, *rItemSet); } void SfxDocumentPage::EnableUseUserData() { bEnableUseUserData = true; m_xUseUserDataCB->show(); m_xDeleteBtn->show(); } bool SfxDocumentPage::FillItemSet( SfxItemSet* rSet ) { bool bRet = false; if ( !bHandleDelete && bEnableUseUserData && m_xUseUserDataCB->get_state_changed_from_saved() ) { const SfxItemSet* pExpSet = GetDialogExampleSet(); const SfxDocumentInfoItem* pInfoItem; if ( pExpSet && (pInfoItem = pExpSet->GetItemIfSet( SID_DOCINFO ) ) ) { bool bUseData = ( TRISTATE_TRUE == m_xUseUserDataCB->get_state() ); const_cast(pInfoItem)->SetUseUserData( bUseData ); rSet->Put( *pInfoItem ); bRet = true; } } if ( bHandleDelete ) { const SfxItemSet* pExpSet = GetDialogExampleSet(); const SfxDocumentInfoItem* pInfoItem; if ( pExpSet && (pInfoItem = pExpSet->GetItemIfSet( SID_DOCINFO )) ) { bool bUseAuthor = bEnableUseUserData && m_xUseUserDataCB->get_active(); SfxDocumentInfoItem newItem( *pInfoItem ); newItem.resetUserData( bUseAuthor ? SvtUserOptions().GetFullName() : OUString() ); const_cast(pInfoItem)->SetUseUserData( TRISTATE_TRUE == m_xUseUserDataCB->get_state() ); newItem.SetUseUserData( TRISTATE_TRUE == m_xUseUserDataCB->get_state() ); newItem.SetDeleteUserData( true ); rSet->Put( newItem ); bRet = true; } } if ( m_xUseThumbnailSaveCB->get_state_changed_from_saved() ) { const SfxItemSet* pExpSet = GetDialogExampleSet(); const SfxDocumentInfoItem* pInfoItem; if ( pExpSet && (pInfoItem = pExpSet->GetItemIfSet( SID_DOCINFO )) ) { bool bUseThumbnail = ( TRISTATE_TRUE == m_xUseThumbnailSaveCB->get_state() ); const_cast(pInfoItem)->SetUseThumbnailSave( bUseThumbnail ); rSet->Put( *pInfoItem ); bRet = true; } } SfxObjectShell* pDocSh = SfxObjectShell::Current(); if (pDocSh) { uno::Reference xFac(pDocSh->GetModel(), uno::UNO_QUERY); if (xFac.is()) { uno::Reference xProps(xFac->createInstance(u"com.sun.star.document.Settings"_ustr), uno::UNO_QUERY); if (xProps.is()) { sal_Int32 nImagePreferredDPI = 0; if (m_xImagePreferredDpiCheckButton->get_state() == TRISTATE_TRUE) { OUString aImagePreferredDPIString = m_xImagePreferredDpiComboBox->get_active_text(); nImagePreferredDPI = aImagePreferredDPIString.toInt32(); } xProps->setPropertyValue(u"ImagePreferredDPI"_ustr, uno::Any(nImagePreferredDPI)); } } } return bRet; } void SfxDocumentPage::Reset( const SfxItemSet* rSet ) { // Determine the document information const SfxDocumentInfoItem& rInfoItem = rSet->Get(SID_DOCINFO); // template data if (rInfoItem.HasTemplate()) { const OUString& rName = rInfoItem.getTemplateName(); if (rName.getLength() > SAL_MAX_INT16) // tdf#122780 pick some ~arbitrary max size m_xTemplValFt->set_label(rName.copy(0, SAL_MAX_INT16)); else m_xTemplValFt->set_label(rName); } else { m_xTemplFt->hide(); m_xTemplValFt->hide(); } // determine file name OUString aFile( rInfoItem.GetValue() ); OUString aFactory( aFile ); if ( aFile.getLength() > 2 && aFile[0] == '[' ) { sal_Int32 nPos = aFile.indexOf( ']' ); aFactory = aFile.copy( 1, nPos-1 ); aFile = aFile.copy( nPos+1 ); } // determine name INetURLObject aURL(aFile); OUString aName = aURL.GetLastName(INetURLObject::DecodeMechanism::WithCharset); if ( aName.isEmpty() || aURL.GetProtocol() == INetProtocol::PrivSoffice ) aName = SfxResId( STR_NONAME ); m_xNameED->set_label( aName ); // determine context symbol aURL.SetSmartProtocol( INetProtocol::File ); aURL.SetSmartURL( aFactory); const OUString& rMainURL = aURL.GetMainURL( INetURLObject::DecodeMechanism::NONE ); OUString aImage = SvFileInformationManager::GetImageId( aURL, true ); m_xBmp->set_from_icon_name(aImage); // determine size and type OUString aSizeText( m_aUnknownSize ); // we might already know the size as an optional argument passed to .uno:SetDocumentProperties sal_Int64 nSize = rInfoItem.getFileSize(); // otherwise, for some protocols we can reliably query for it if (nSize == -1 && (aURL.GetProtocol() == INetProtocol::File || aURL.isAnyKnownWebDAVScheme())) nSize = SfxContentHelper::GetSize( aURL.GetMainURL( INetURLObject::DecodeMechanism::NONE ) ); if (nSize != -1) aSizeText = CreateSizeText( nSize ); m_xShowSizeFT->set_label( aSizeText ); OUString aDescription = SvFileInformationManager::GetDescription( INetURLObject(rMainURL) ); if ( aDescription.isEmpty() ) aDescription = SfxResId( STR_SFX_NEWOFFICEDOC ); m_xShowTypeFT->set_label( aDescription ); // determine location // online we don't know file location so we just set it as the name if (comphelper::LibreOfficeKit::isActive()) { m_xFileValEd->set_label(aName); m_xFileValEd->set_uri(aName); // Disable setting/changing password on text files. // Perhaps this needs to be done for both Online and Desktop. OUString sExtension(INetURLObject(rMainURL).getExtension()); if (!sExtension.isEmpty()) { sExtension = sExtension.toAsciiLowerCase(); if (sExtension == "txt" || sExtension == "csv") m_xChangePassBtn->set_sensitive(false); } } else { aURL.SetSmartURL( aFile); if ( aURL.GetProtocol() == INetProtocol::File ) { INetURLObject aPath(std::move(aURL)); aPath.setFinalSlash(); aPath.removeSegment(); // we know it's a folder -> don't need the final slash, but it's better for WB_PATHELLIPSIS aPath.removeFinalSlash(); OUString aText( aPath.PathToFileName() ); //! (pb) MaxLen? m_xFileValEd->set_label(aText); OUString aURLStr; osl::FileBase::getFileURLFromSystemPath(aText, aURLStr); m_xFileValEd->set_uri(aURLStr); } else if (aURL.GetProtocol() != INetProtocol::PrivSoffice) { m_xFileValEd->set_label(aURL.GetPartBeforeLastName()); m_xFileValEd->set_uri(m_xFileValEd->get_label()); } } // handle access data bool bUseUserData = rInfoItem.IsUseUserData(); const LocaleDataWrapper& rLocaleWrapper( Application::GetSettings().GetLocaleDataWrapper() ); m_xCreateValFt->set_label( ConvertDateTime_Impl( rInfoItem.getAuthor(), rInfoItem.getCreationDate(), rLocaleWrapper ) ); util::DateTime aTime( rInfoItem.getModificationDate() ); if ( aTime.Month > 0 ) m_xChangeValFt->set_label( ConvertDateTime_Impl( rInfoItem.getModifiedBy(), aTime, rLocaleWrapper ) ); aTime = rInfoItem.getPrintDate(); if ( aTime.Month > 0 ) m_xPrintValFt->set_label( ConvertDateTime_Impl( rInfoItem.getPrintedBy(), aTime, rLocaleWrapper ) ); const tools::Long nTime = rInfoItem.getEditingDuration(); if ( bUseUserData ) { assert(SAL_MIN_INT32 <= nTime/86400 && nTime/86400 <= SAL_MAX_INT32); const tools::Duration aD( static_cast(nTime)/86400, (nTime%86400)/3600, (nTime%3600)/60, nTime%60, 0); m_xTimeLogValFt->set_label( rLocaleWrapper.getDuration( aD ) ); m_xDocNoValFt->set_label( OUString::number( rInfoItem.getEditingCycles() ) ); } bool bUseThumbnailSave = rInfoItem.IsUseThumbnailSave(); // Check for cmis properties where otherwise unavailable if ( rInfoItem.isCmisDocument( ) ) { const uno::Sequence< document::CmisProperty > aCmisProps = rInfoItem.GetCmisProperties(); for ( const auto& rCmisProp : aCmisProps ) { if ( rCmisProp.Id == "cmis:contentStreamLength" && aSizeText == m_aUnknownSize ) { Sequence< sal_Int64 > seqValue; rCmisProp.Value >>= seqValue; SvNumberFormatter aNumberFormatter( ::comphelper::getProcessComponentContext(), Application::GetSettings().GetLanguageTag().getLanguageType() ); sal_uInt32 nIndex = aNumberFormatter.GetFormatIndex( NF_NUMBER_SYSTEM ); if ( seqValue.hasElements() ) { OUString sValue; aNumberFormatter.GetInputLineString( seqValue[0], nIndex, sValue ); m_xShowSizeFT->set_label( CreateSizeText( sValue.toInt64( ) ) ); } } util::DateTime uDT; OUString emptyDate = ConvertDateTime_Impl( u"", uDT, rLocaleWrapper ); if ( rCmisProp.Id == "cmis:creationDate" && (m_xCreateValFt->get_label() == emptyDate || m_xCreateValFt->get_label().isEmpty())) { Sequence< util::DateTime > seqValue; rCmisProp.Value >>= seqValue; if ( seqValue.hasElements() ) { m_xCreateValFt->set_label( ConvertDateTime_Impl( u"", seqValue[0], rLocaleWrapper ) ); } } if ( rCmisProp.Id == "cmis:lastModificationDate" && (m_xChangeValFt->get_label() == emptyDate || m_xChangeValFt->get_label().isEmpty())) { Sequence< util::DateTime > seqValue; rCmisProp.Value >>= seqValue; if ( seqValue.hasElements() ) { m_xChangeValFt->set_label( ConvertDateTime_Impl( u"", seqValue[0], rLocaleWrapper ) ); } } } } m_xUseUserDataCB->set_active(bUseUserData); m_xUseUserDataCB->save_state(); m_xUseUserDataCB->set_sensitive( bEnableUseUserData ); bHandleDelete = false; m_xDeleteBtn->set_sensitive( bEnableUseUserData ); m_xUseThumbnailSaveCB->set_active(bUseThumbnailSave); m_xUseThumbnailSaveCB->save_state(); SfxObjectShell* pDocSh = SfxObjectShell::Current(); sal_Int32 nImagePreferredDPI = 0; if (pDocSh) { try { uno::Reference< lang::XMultiServiceFactory > xFac( pDocSh->GetModel(), uno::UNO_QUERY ); if (xFac) { uno::Reference< beans::XPropertySet > xProps( xFac->createInstance(u"com.sun.star.document.Settings"_ustr), uno::UNO_QUERY ); if (xProps) xProps->getPropertyValue(u"ImagePreferredDPI"_ustr) >>= nImagePreferredDPI; } } catch( uno::Exception& ) { } } if (nImagePreferredDPI > 0) { m_xImagePreferredDpiCheckButton->set_state(TRISTATE_TRUE); m_xImagePreferredDpiComboBox->set_sensitive(true); m_xImagePreferredDpiComboBox->set_entry_text(OUString::number(nImagePreferredDPI)); } else { m_xImagePreferredDpiCheckButton->set_state(TRISTATE_FALSE); m_xImagePreferredDpiComboBox->set_sensitive(false); m_xImagePreferredDpiComboBox->set_entry_text(u""_ustr); } } SfxDocumentInfoDialog::SfxDocumentInfoDialog(weld::Window* pParent, const SfxItemSet& rItemSet) : SfxTabDialogController(pParent, u"sfx/ui/documentpropertiesdialog.ui"_ustr, u"DocumentPropertiesDialog"_ustr, &rItemSet) { const SfxDocumentInfoItem& rInfoItem = rItemSet.Get( SID_DOCINFO ); #ifdef DBG_UTIL const SfxStringItem* pURLItem = rItemSet.GetItem(SID_BASEURL, false); DBG_ASSERT( pURLItem, "No BaseURL provided for InternetTabPage!" ); #endif // Determine the Titles OUString aTitle(m_xDialog->get_title()); const SfxStringItem* pItem = rItemSet.GetItemIfSet( SID_EXPLORER_PROPS_START, false ); if ( !pItem ) { // File name const OUString& aFile( rInfoItem.GetValue() ); INetURLObject aURL; aURL.SetSmartProtocol( INetProtocol::File ); aURL.SetSmartURL( aFile); if ( INetProtocol::PrivSoffice != aURL.GetProtocol() ) { OUString aLastName( aURL.GetLastName() ); if ( !aLastName.isEmpty() ) aTitle = aTitle.replaceFirst("%1", aLastName); else aTitle = aTitle.replaceFirst("%1", aFile); } else aTitle = aTitle.replaceFirst("%1", SfxResId( STR_NONAME )); } else { aTitle = aTitle.replaceFirst("%1", pItem->GetValue()); } m_xDialog->set_title(aTitle); // Property Pages AddTabPage(u"general"_ustr, SfxDocumentPage::Create, nullptr); AddTabPage(u"description"_ustr, SfxDocumentDescPage::Create, nullptr); AddTabPage(u"customprops"_ustr, SfxCustomPropertiesPage::Create, nullptr); if (rInfoItem.isCmisDocument()) AddTabPage(u"cmisprops"_ustr, SfxCmisPropertiesPage::Create, nullptr); else RemoveTabPage(u"cmisprops"_ustr); // Disable security page for online as not fully asynced yet if (!comphelper::LibreOfficeKit::isActive()) AddTabPage(u"security"_ustr, SfxSecurityPage::Create, nullptr); else RemoveTabPage(u"security"_ustr); } void SfxDocumentInfoDialog::PageCreated(const OUString& rId, SfxTabPage &rPage) { if (rId == "general") static_cast(rPage).EnableUseUserData(); } void SfxDocumentInfoDialog::AddFontTabPage() { AddTabPage(u"font"_ustr, SfxResId(STR_FONT_TABPAGE), SfxDocumentFontsPage::Create); } // class CustomPropertiesYesNoButton ------------------------------------- CustomPropertiesYesNoButton::CustomPropertiesYesNoButton(std::unique_ptr xTopLevel, std::unique_ptr xYesButton, std::unique_ptr xNoButton) : m_xTopLevel(std::move(xTopLevel)) , m_xYesButton(std::move(xYesButton)) , m_xNoButton(std::move(xNoButton)) { CheckNo(); } CustomPropertiesYesNoButton::~CustomPropertiesYesNoButton() { } DurationDialog_Impl::DurationDialog_Impl(weld::Widget* pParent, const util::Duration& rDuration) : GenericDialogController(pParent, u"sfx/ui/editdurationdialog.ui"_ustr, u"EditDurationDialog"_ustr) , m_xNegativeCB(m_xBuilder->weld_check_button(u"negative"_ustr)) , m_xYearNF(m_xBuilder->weld_spin_button(u"years"_ustr)) , m_xMonthNF(m_xBuilder->weld_spin_button(u"months"_ustr)) , m_xDayNF(m_xBuilder->weld_spin_button(u"days"_ustr)) , m_xHourNF(m_xBuilder->weld_spin_button(u"hours"_ustr)) , m_xMinuteNF(m_xBuilder->weld_spin_button(u"minutes"_ustr)) , m_xSecondNF(m_xBuilder->weld_spin_button(u"seconds"_ustr)) , m_xMSecondNF(m_xBuilder->weld_spin_button(u"milliseconds"_ustr)) { m_xNegativeCB->set_active(rDuration.Negative); m_xYearNF->set_value(rDuration.Years); m_xMonthNF->set_value(rDuration.Months); m_xDayNF->set_value(rDuration.Days); m_xHourNF->set_value(rDuration.Hours); m_xMinuteNF->set_value(rDuration.Minutes); m_xSecondNF->set_value(rDuration.Seconds); m_xMSecondNF->set_value(rDuration.NanoSeconds); } util::Duration DurationDialog_Impl::GetDuration() const { util::Duration aRet; aRet.Negative = m_xNegativeCB->get_active(); aRet.Years = m_xYearNF->get_value(); aRet.Months = m_xMonthNF->get_value(); aRet.Days = m_xDayNF->get_value(); aRet.Hours = m_xHourNF->get_value(); aRet.Minutes = m_xMinuteNF->get_value(); aRet.Seconds = m_xSecondNF->get_value(); aRet.NanoSeconds = m_xMSecondNF->get_value(); return aRet; } CustomPropertiesDurationField::CustomPropertiesDurationField(std::unique_ptr xEntry, std::unique_ptr xEditButton) : m_xEntry(std::move(xEntry)) , m_xEditButton(std::move(xEditButton)) { m_xEditButton->connect_clicked(LINK(this, CustomPropertiesDurationField, ClickHdl)); SetDuration( util::Duration(false, 0, 0, 0, 0, 0, 0, 0) ); } void CustomPropertiesDurationField::set_visible(bool bVisible) { m_xEntry->set_visible(bVisible); m_xEditButton->set_visible(bVisible); } void CustomPropertiesDurationField::SetDuration( const util::Duration& rDuration ) { m_aDuration = rDuration; OUString sText = (rDuration.Negative ? OUString('-') : OUString('+')) + SfxResId(SFX_ST_DURATION_FORMAT); sText = sText.replaceFirst( "%1", OUString::number( rDuration.Years ) ); sText = sText.replaceFirst( "%2", OUString::number( rDuration.Months ) ); sText = sText.replaceFirst( "%3", OUString::number( rDuration.Days ) ); sText = sText.replaceFirst( "%4", OUString::number( rDuration.Hours ) ); sText = sText.replaceFirst( "%5", OUString::number( rDuration.Minutes) ); sText = sText.replaceFirst( "%6", OUString::number( rDuration.Seconds) ); m_xEntry->set_text(sText); } IMPL_LINK(CustomPropertiesDurationField, ClickHdl, weld::Button&, rButton, void) { m_xDurationDialog = std::make_shared(&rButton, GetDuration()); weld::DialogController::runAsync(m_xDurationDialog, [&](sal_Int32 response) { if (response == RET_OK) { SetDuration(m_xDurationDialog->GetDuration()); } }); } CustomPropertiesDurationField::~CustomPropertiesDurationField() { if (m_xDurationDialog) m_xDurationDialog->response(RET_CANCEL); } namespace { void fillNameBox(weld::ComboBox& rNameBox) { for (size_t i = 0; i < SAL_N_ELEMENTS(SFX_CB_PROPERTY_STRINGARRAY); ++i) rNameBox.append_text(SfxResId(SFX_CB_PROPERTY_STRINGARRAY[i])); Size aSize(rNameBox.get_preferred_size()); rNameBox.set_size_request(aSize.Width(), aSize.Height()); } void fillTypeBox(weld::ComboBox& rTypeBox) { for (size_t i = 0; i < SAL_N_ELEMENTS(SFX_LB_PROPERTY_STRINGARRAY); ++i) { OUString sId(OUString::number(SFX_LB_PROPERTY_STRINGARRAY[i].second)); rTypeBox.append(sId, SfxResId(SFX_LB_PROPERTY_STRINGARRAY[i].first)); } rTypeBox.set_active(0); Size aSize(rTypeBox.get_preferred_size()); rTypeBox.set_size_request(aSize.Width(), aSize.Height()); } } // struct CustomPropertyLine --------------------------------------------- CustomPropertyLine::CustomPropertyLine(CustomPropertiesWindow* pParent, weld::Widget* pContainer) : m_pParent(pParent) , m_xBuilder(Application::CreateBuilder(pContainer, u"sfx/ui/linefragment.ui"_ustr)) , m_xLine(m_xBuilder->weld_container(u"lineentry"_ustr)) , m_xNameBox(m_xBuilder->weld_combo_box(u"namebox"_ustr)) , m_xTypeBox(m_xBuilder->weld_combo_box(u"typebox"_ustr)) , m_xValueEdit(m_xBuilder->weld_entry(u"valueedit"_ustr)) , m_xDateTimeBox(m_xBuilder->weld_widget(u"datetimebox"_ustr)) , m_xDateField(new CustomPropertiesDateField(new SvtCalendarBox(m_xBuilder->weld_menu_button(u"date"_ustr)))) , m_xTimeField(new CustomPropertiesTimeField(m_xBuilder->weld_formatted_spin_button(u"time"_ustr))) , m_xDurationBox(m_xBuilder->weld_widget(u"durationbox"_ustr)) , m_xDurationField(new CustomPropertiesDurationField(m_xBuilder->weld_entry(u"duration"_ustr), m_xBuilder->weld_button(u"durationbutton"_ustr))) , m_xYesNoButton(new CustomPropertiesYesNoButton(m_xBuilder->weld_widget(u"yesno"_ustr), m_xBuilder->weld_radio_button(u"yes"_ustr), m_xBuilder->weld_radio_button(u"no"_ustr))) , m_xRemoveButton(m_xBuilder->weld_button(u"remove"_ustr)) , m_bTypeLostFocus( false ) { fillNameBox(*m_xNameBox); fillTypeBox(*m_xTypeBox); m_xTypeBox->connect_changed(LINK(this, CustomPropertyLine, TypeHdl)); m_xRemoveButton->connect_clicked(LINK(this, CustomPropertyLine, RemoveHdl)); m_xValueEdit->connect_focus_out(LINK(this, CustomPropertyLine, EditLoseFocusHdl)); //add lose focus handlers of date/time fields m_xTypeBox->connect_focus_out(LINK(this, CustomPropertyLine, BoxLoseFocusHdl)); } void CustomPropertyLine::Clear() { m_xNameBox->set_active(-1); m_xValueEdit->set_text(OUString()); } void CustomPropertyLine::Hide() { m_xLine->hide(); } CustomPropertiesWindow::CustomPropertiesWindow(weld::Container& rParent, weld::Label& rHeaderAccName, weld::Label& rHeaderAccType, weld::Label& rHeaderAccValue) : m_nHeight(0) , m_nLineHeight(0) , m_nScrollPos(0) , m_pCurrentLine(nullptr) , m_aNumberFormatter(::comphelper::getProcessComponentContext(), Application::GetSettings().GetLanguageTag().getLanguageType()) , m_aEditLoseFocusIdle("sfx2 CustomPropertiesWindow loseFocusIdle") , m_aBoxLoseFocusIdle("sfx2 CustomPropertiesWindow m_aBoxLoseFocusIdle") , m_rBody(rParent) , m_rHeaderAccName(rHeaderAccName) , m_rHeaderAccType(rHeaderAccType) , m_rHeaderAccValue(rHeaderAccValue) { m_aEditLoseFocusIdle.SetPriority( TaskPriority::LOWEST ); m_aEditLoseFocusIdle.SetInvokeHandler( LINK( this, CustomPropertiesWindow, EditTimeoutHdl ) ); m_aBoxLoseFocusIdle.SetPriority( TaskPriority::LOWEST ); m_aBoxLoseFocusIdle.SetInvokeHandler( LINK( this, CustomPropertiesWindow, BoxTimeoutHdl ) ); } CustomPropertiesWindow::~CustomPropertiesWindow() { m_aEditLoseFocusIdle.Stop(); m_aBoxLoseFocusIdle.Stop(); m_pCurrentLine = nullptr; } void CustomPropertyLine::DoTypeHdl(const weld::ComboBox& rBox) { auto nType = rBox.get_active_id().toInt32(); m_xValueEdit->set_visible( (Custom_Type_Text == nType) || (Custom_Type_Number == nType) ); m_xDateTimeBox->set_visible( (Custom_Type_Date == nType) || (Custom_Type_Datetime == nType) ); m_xDateField->set_visible( (Custom_Type_Date == nType) || (Custom_Type_Datetime == nType) ); m_xTimeField->set_visible( Custom_Type_Datetime == nType ); m_xDurationBox->set_visible( Custom_Type_Duration == nType ); m_xDurationField->set_visible( Custom_Type_Duration == nType ); m_xYesNoButton->set_visible( Custom_Type_Boolean == nType ); } IMPL_LINK(CustomPropertyLine, TypeHdl, weld::ComboBox&, rBox, void) { DoTypeHdl(rBox); } void CustomPropertiesWindow::Remove(const CustomPropertyLine* pLine) { StoreCustomProperties(); auto pFound = std::find_if( m_aCustomPropertiesLines.begin(), m_aCustomPropertiesLines.end(), [&] (const std::unique_ptr& p) { return p.get() == pLine; }); if ( pFound != m_aCustomPropertiesLines.end() ) { sal_uInt32 nLineNumber = pFound - m_aCustomPropertiesLines.begin(); sal_uInt32 nDataModelIndex = GetCurrentDataModelPosition() + nLineNumber; m_aCustomProperties.erase(m_aCustomProperties.begin() + nDataModelIndex); ReloadLinesContent(); } m_aRemovedHdl.Call(nullptr); } IMPL_LINK_NOARG(CustomPropertyLine, RemoveHdl, weld::Button&, void) { m_pParent->Remove(this); } void CustomPropertiesWindow::EditLoseFocus(CustomPropertyLine* pLine) { m_pCurrentLine = pLine; m_aEditLoseFocusIdle.Start(); } IMPL_LINK_NOARG(CustomPropertyLine, EditLoseFocusHdl, weld::Widget&, void) { if (!m_bTypeLostFocus) m_pParent->EditLoseFocus(this); else m_bTypeLostFocus = false; } void CustomPropertiesWindow::BoxLoseFocus(CustomPropertyLine* pLine) { m_pCurrentLine = pLine; m_aBoxLoseFocusIdle.Start(); } IMPL_LINK_NOARG(CustomPropertyLine, BoxLoseFocusHdl, weld::Widget&, void) { m_pParent->BoxLoseFocus(this); } IMPL_LINK_NOARG(CustomPropertiesWindow, EditTimeoutHdl, Timer *, void) { ValidateLine( m_pCurrentLine, false ); } IMPL_LINK_NOARG(CustomPropertiesWindow, BoxTimeoutHdl, Timer *, void) { ValidateLine( m_pCurrentLine, true ); } bool CustomPropertiesWindow::IsLineValid( CustomPropertyLine* pLine ) const { bool bIsValid = true; pLine->m_bTypeLostFocus = false; auto nType = pLine->m_xTypeBox->get_active_id().toInt32(); OUString sValue = pLine->m_xValueEdit->get_text(); if ( sValue.isEmpty() ) return true; sal_uInt32 nIndex = NUMBERFORMAT_ENTRY_NOT_FOUND; if ( Custom_Type_Number == nType ) // tdf#116214 Scientific format allows to use also standard numbers nIndex = const_cast< SvNumberFormatter& >( m_aNumberFormatter ).GetFormatIndex( NF_SCIENTIFIC_000E00 ); else if ( Custom_Type_Date == nType ) nIndex = const_cast< SvNumberFormatter& >( m_aNumberFormatter).GetFormatIndex( NF_DATE_SYS_DDMMYYYY ); if ( nIndex != NUMBERFORMAT_ENTRY_NOT_FOUND ) { sal_uInt32 nTemp = nIndex; double fDummy = 0.0; bIsValid = const_cast< SvNumberFormatter& >( m_aNumberFormatter ).IsNumberFormat( sValue, nIndex, fDummy ); if ( bIsValid && nTemp != nIndex ) // sValue is a number but the format doesn't match the index bIsValid = false; } return bIsValid; } void CustomPropertiesWindow::ValidateLine( CustomPropertyLine* pLine, bool bIsFromTypeBox ) { if (pLine && !IsLineValid(pLine)) { if ( bIsFromTypeBox ) // LoseFocus of TypeBox pLine->m_bTypeLostFocus = true; std::unique_ptr xMessageBox(Application::CreateMessageDialog(&m_rBody, VclMessageType::Question, VclButtonsType::OkCancel, SfxResId(STR_SFX_QUERY_WRONG_TYPE))); if (xMessageBox->run() == RET_OK) pLine->m_xTypeBox->set_active_id(OUString::number(Custom_Type_Text)); else pLine->m_xValueEdit->grab_focus(); } } void CustomPropertiesWindow::SetVisibleLineCount(sal_uInt32 nCount) { while (GetExistingLineCount() < nCount) { CreateNewLine(); } } void CustomPropertiesWindow::AddLine(const OUString& sName, Any const & rAny) { m_aCustomProperties.push_back(std::unique_ptr(new CustomProperty(sName, rAny))); ReloadLinesContent(); } void CustomPropertiesWindow::CreateNewLine() { CustomPropertyLine* pNewLine = new CustomPropertyLine(this, &m_rBody); pNewLine->m_xNameBox->set_accessible_relation_labeled_by(&m_rHeaderAccName); pNewLine->m_xNameBox->set_accessible_name(m_rHeaderAccName.get_label()); pNewLine->m_xTypeBox->set_accessible_relation_labeled_by(&m_rHeaderAccType); pNewLine->m_xTypeBox->set_accessible_name(m_rHeaderAccType.get_label()); pNewLine->m_xValueEdit->set_accessible_relation_labeled_by(&m_rHeaderAccValue); pNewLine->m_xValueEdit->set_accessible_name(m_rHeaderAccValue.get_label()); m_aCustomPropertiesLines.emplace_back( pNewLine ); // this breaks online's jsdialogbuilder if (!comphelper::LibreOfficeKit::isActive()){ // for ui-testing. Distinguish the elements in the lines sal_uInt16 nSize = m_aCustomPropertiesLines.size(); pNewLine->m_xNameBox->set_buildable_name( pNewLine->m_xNameBox->get_buildable_name() + OUString::number(nSize)); pNewLine->m_xTypeBox->set_buildable_name( pNewLine->m_xTypeBox->get_buildable_name() + OUString::number(nSize)); pNewLine->m_xValueEdit->set_buildable_name( pNewLine->m_xValueEdit->get_buildable_name() + OUString::number(nSize)); pNewLine->m_xRemoveButton->set_buildable_name( pNewLine->m_xRemoveButton->get_buildable_name() + OUString::number(nSize)); } pNewLine->DoTypeHdl(*pNewLine->m_xTypeBox); } bool CustomPropertiesWindow::AreAllLinesValid() const { bool bRet = true; for ( std::unique_ptr const & pLine : m_aCustomPropertiesLines ) { if ( !IsLineValid( pLine.get() ) ) { bRet = false; break; } } return bRet; } void CustomPropertiesWindow::ClearAllLines() { for (auto& pLine : m_aCustomPropertiesLines) { pLine->Clear(); } m_pCurrentLine = nullptr; m_aCustomProperties.clear(); m_nScrollPos = 0; } void CustomPropertiesWindow::DoScroll( sal_Int32 nNewPos ) { StoreCustomProperties(); m_nScrollPos += nNewPos; ReloadLinesContent(); } Sequence< beans::PropertyValue > CustomPropertiesWindow::GetCustomProperties() { StoreCustomProperties(); Sequence< beans::PropertyValue > aPropertiesSeq(GetTotalLineCount()); std::transform( m_aCustomProperties.begin(), m_aCustomProperties.end(), aPropertiesSeq.getArray(), [](const auto& el) { return comphelper::makePropertyValue(el->m_sName, el->m_aValue); }); return aPropertiesSeq; } CustomPropertiesTimeField::CustomPropertiesTimeField(std::unique_ptr xTimeField) : m_xTimeField(std::move(xTimeField)) , m_xFormatter(new weld::TimeFormatter(*m_xTimeField)) , m_isUTC(false) { m_xFormatter->SetExtFormat(ExtTimeFieldFormat::LongDuration); m_xFormatter->EnableEmptyField(false); } tools::Time CustomPropertiesTimeField::get_value() const { return m_xFormatter->GetTime(); } void CustomPropertiesTimeField::set_value(const tools::Time& rTime) { m_xFormatter->SetTime(rTime); } CustomPropertiesTimeField::~CustomPropertiesTimeField() { } CustomPropertiesDateField::CustomPropertiesDateField(SvtCalendarBox* pDateField) : m_xDateField(pDateField) { DateTime aDateTime(DateTime::SYSTEM); m_xDateField->set_date(aDateTime); } void CustomPropertiesDateField::set_visible(bool bVisible) { m_xDateField->set_visible(bVisible); } Date CustomPropertiesDateField::get_date() const { return m_xDateField->get_date(); } void CustomPropertiesDateField::set_date(const Date& rDate) { m_xDateField->set_date(rDate); } CustomPropertiesDateField::~CustomPropertiesDateField() { } void CustomPropertiesWindow::StoreCustomProperties() { sal_uInt32 nDataModelPos = GetCurrentDataModelPosition(); for (sal_uInt32 i = 0; nDataModelPos + i < GetTotalLineCount() && i < GetExistingLineCount(); i++) { CustomPropertyLine* pLine = m_aCustomPropertiesLines[i].get(); OUString sPropertyName = pLine->m_xNameBox->get_active_text(); if (!sPropertyName.isEmpty()) { m_aCustomProperties[nDataModelPos + i]->m_sName = sPropertyName; auto nType = pLine->m_xTypeBox->get_active_id().toInt32(); if (Custom_Type_Number == nType) { double nValue = 0; sal_uInt32 nIndex = m_aNumberFormatter.GetFormatIndex(NF_NUMBER_SYSTEM); bool bIsNum = m_aNumberFormatter. IsNumberFormat(pLine->m_xValueEdit->get_text(), nIndex, nValue); if (bIsNum) m_aCustomProperties[nDataModelPos + i]->m_aValue <<= nValue; } else if (Custom_Type_Boolean == nType) { bool bValue = pLine->m_xYesNoButton->IsYesChecked(); m_aCustomProperties[nDataModelPos + i]->m_aValue <<= bValue; } else if (Custom_Type_Datetime == nType) { Date aTmpDate = pLine->m_xDateField->get_date(); tools::Time aTmpTime = pLine->m_xTimeField->get_value(); util::DateTime const aDateTime(aTmpTime.GetNanoSec(), aTmpTime.GetSec(), aTmpTime.GetMin(), aTmpTime.GetHour(), aTmpDate.GetDay(), aTmpDate.GetMonth(), aTmpDate.GetYear(), pLine->m_xTimeField->m_isUTC); if (pLine->m_xDateField->m_TZ) { m_aCustomProperties[nDataModelPos + i]->m_aValue <<= util::DateTimeWithTimezone( aDateTime, *pLine->m_xDateField->m_TZ); } else { m_aCustomProperties[nDataModelPos + i]->m_aValue <<= aDateTime; } } else if (Custom_Type_Date == nType) { Date aTmpDate = pLine->m_xDateField->get_date(); util::Date const aDate(aTmpDate.GetDay(), aTmpDate.GetMonth(), aTmpDate.GetYear()); if (pLine->m_xDateField->m_TZ) { m_aCustomProperties[nDataModelPos + i]->m_aValue <<= util::DateWithTimezone( aDate, *pLine->m_xDateField->m_TZ); } else { m_aCustomProperties[nDataModelPos + i]->m_aValue <<= aDate; } } else if (Custom_Type_Duration == nType) { m_aCustomProperties[nDataModelPos + i]->m_aValue <<= pLine->m_xDurationField->GetDuration(); } else { OUString sValue(pLine->m_xValueEdit->get_text()); m_aCustomProperties[nDataModelPos + i]->m_aValue <<= sValue; } } } } void CustomPropertiesWindow::SetCustomProperties(std::vector< std::unique_ptr >&& rProperties) { m_aCustomProperties = std::move(rProperties); ReloadLinesContent(); } void CustomPropertiesWindow::ReloadLinesContent() { double nTmpValue = 0; bool bTmpValue = false; OUString sTmpValue; util::DateTime aTmpDateTime; util::Date aTmpDate; util::DateTimeWithTimezone aTmpDateTimeTZ; util::DateWithTimezone aTmpDateTZ; util::Duration aTmpDuration; SvtSysLocale aSysLocale; const LocaleDataWrapper& rLocaleWrapper = aSysLocale.GetLocaleData(); CustomProperties nType = Custom_Type_Unknown; OUString sValue; sal_uInt32 nDataModelPos = GetCurrentDataModelPosition(); sal_uInt32 i = 0; for (; nDataModelPos + i < GetTotalLineCount() && i < GetExistingLineCount(); i++) { const OUString& rName = m_aCustomProperties[nDataModelPos + i]->m_sName; const css::uno::Any& rAny = m_aCustomProperties[nDataModelPos + i]->m_aValue; CustomPropertyLine* pLine = m_aCustomPropertiesLines[i].get(); pLine->Clear(); pLine->m_xNameBox->set_entry_text(rName); pLine->m_xLine->show(); if (!rAny.hasValue()) { pLine->m_xValueEdit->set_text(OUString()); } else if (rAny >>= nTmpValue) { sal_uInt32 nIndex = m_aNumberFormatter.GetFormatIndex(NF_NUMBER_SYSTEM); m_aNumberFormatter.GetInputLineString(nTmpValue, nIndex, sValue); pLine->m_xValueEdit->set_text(sValue); nType = Custom_Type_Number; } else if (rAny >>= bTmpValue) { sValue = (bTmpValue ? rLocaleWrapper.getTrueWord() : rLocaleWrapper.getFalseWord()); nType = Custom_Type_Boolean; } else if (rAny >>= sTmpValue) { pLine->m_xValueEdit->set_text(sTmpValue); nType = Custom_Type_Text; } else if (rAny >>= aTmpDate) { pLine->m_xDateField->set_date(Date(aTmpDate)); nType = Custom_Type_Date; } else if (rAny >>= aTmpDateTime) { pLine->m_xDateField->set_date(Date(aTmpDateTime)); pLine->m_xTimeField->set_value(tools::Time(aTmpDateTime)); pLine->m_xTimeField->m_isUTC = aTmpDateTime.IsUTC; nType = Custom_Type_Datetime; } else if (rAny >>= aTmpDateTZ) { pLine->m_xDateField->set_date(Date(aTmpDateTZ.DateInTZ.Day, aTmpDateTZ.DateInTZ.Month, aTmpDateTZ.DateInTZ.Year)); pLine->m_xDateField->m_TZ = aTmpDateTZ.Timezone; nType = Custom_Type_Date; } else if (rAny >>= aTmpDateTimeTZ) { util::DateTime const& rDT(aTmpDateTimeTZ.DateTimeInTZ); pLine->m_xDateField->set_date(Date(rDT)); pLine->m_xTimeField->set_value(tools::Time(rDT)); pLine->m_xTimeField->m_isUTC = rDT.IsUTC; pLine->m_xDateField->m_TZ = aTmpDateTimeTZ.Timezone; nType = Custom_Type_Datetime; } else if (rAny >>= aTmpDuration) { nType = Custom_Type_Duration; pLine->m_xDurationField->SetDuration(aTmpDuration); } if (Custom_Type_Boolean == nType) { if (bTmpValue) pLine->m_xYesNoButton->CheckYes(); else pLine->m_xYesNoButton->CheckNo(); } pLine->m_xTypeBox->set_active_id(OUString::number(nType)); pLine->DoTypeHdl(*pLine->m_xTypeBox); } // tdf#132667 - grab focus on the last inserted property if (i > 0 && m_aCustomProperties[nDataModelPos + i - 1]->m_sName.isEmpty()) { CustomPropertyLine* pLine = m_aCustomPropertiesLines[i - 1].get(); pLine->m_xNameBox->grab_focus(); } while (nDataModelPos + i >= GetTotalLineCount() && i < GetExistingLineCount()) { CustomPropertyLine* pLine = m_aCustomPropertiesLines[i].get(); pLine->Hide(); i++; } } CustomPropertiesControl::CustomPropertiesControl() : m_nThumbPos(0) { } void CustomPropertiesControl::Init(weld::Builder& rBuilder) { m_xBox = rBuilder.weld_widget(u"box"_ustr); m_xBody = rBuilder.weld_container(u"properties"_ustr); m_xName = rBuilder.weld_label(u"name"_ustr); m_xType = rBuilder.weld_label(u"type"_ustr); m_xValue = rBuilder.weld_label(u"value"_ustr); m_xVertScroll = rBuilder.weld_scrolled_window(u"scroll"_ustr, true); m_xPropertiesWin.reset(new CustomPropertiesWindow(*m_xBody, *m_xName, *m_xType, *m_xValue)); m_xBox->set_stack_background(); m_xVertScroll->show(); std::unique_ptr xNewLine(new CustomPropertyLine(m_xPropertiesWin.get(), m_xBody.get())); Size aLineSize(xNewLine->m_xLine->get_preferred_size()); m_xPropertiesWin->SetLineHeight(aLineSize.Height() + 6); m_xBody->set_size_request(aLineSize.Width() + 6, -1); auto nHeight = aLineSize.Height() * 8; m_xVertScroll->set_size_request(-1, nHeight + 6); m_xPropertiesWin->SetHeight(nHeight); m_xVertScroll->connect_size_allocate(LINK(this, CustomPropertiesControl, ResizeHdl)); m_xName->set_size_request(xNewLine->m_xNameBox->get_preferred_size().Width(), -1); m_xType->set_size_request(xNewLine->m_xTypeBox->get_preferred_size().Width(), -1); m_xValue->set_size_request(xNewLine->m_xValueEdit->get_preferred_size().Width(), -1); m_xBody->move(xNewLine->m_xLine.get(), nullptr); xNewLine.reset(); m_xPropertiesWin->SetRemovedHdl( LINK( this, CustomPropertiesControl, RemovedHdl ) ); m_xVertScroll->vadjustment_set_lower(0); m_xVertScroll->vadjustment_set_upper(0); m_xVertScroll->vadjustment_set_page_size(0xFFFF); Link aScrollLink = LINK( this, CustomPropertiesControl, ScrollHdl ); m_xVertScroll->connect_vadjustment_changed(aScrollLink); ResizeHdl(Size(-1, nHeight)); } IMPL_LINK(CustomPropertiesControl, ResizeHdl, const Size&, rSize, void) { int nHeight = rSize.Height() - 6; if (nHeight == m_xPropertiesWin->GetHeight()) return; m_xPropertiesWin->SetHeight(nHeight); sal_Int32 nScrollOffset = m_xPropertiesWin->GetLineHeight(); sal_Int32 nVisibleEntries = nHeight / nScrollOffset; m_xPropertiesWin->SetVisibleLineCount( nVisibleEntries ); m_xVertScroll->vadjustment_set_page_increment( nVisibleEntries - 1 ); m_xVertScroll->vadjustment_set_page_size( nVisibleEntries ); m_xPropertiesWin->ReloadLinesContent(); } CustomPropertiesControl::~CustomPropertiesControl() { } IMPL_LINK( CustomPropertiesControl, ScrollHdl, weld::ScrolledWindow&, rScrollBar, void ) { sal_Int32 nOffset = m_xPropertiesWin->GetLineHeight(); int nThumbPos = rScrollBar.vadjustment_get_value(); nOffset *= ( m_nThumbPos - nThumbPos ); m_nThumbPos = nThumbPos; m_xPropertiesWin->DoScroll( nOffset ); } IMPL_LINK_NOARG(CustomPropertiesControl, RemovedHdl, void*, void) { auto nLineCount = m_xPropertiesWin->GetTotalLineCount(); m_xVertScroll->vadjustment_set_upper(nLineCount + 1); if (m_xPropertiesWin->GetTotalLineCount() > m_xPropertiesWin->GetExistingLineCount()) { m_xVertScroll->vadjustment_set_value(nLineCount - 1); ScrollHdl(*m_xVertScroll); } } void CustomPropertiesControl::AddLine( Any const & rAny ) { m_xPropertiesWin->AddLine( OUString(), rAny ); auto nLineCount = m_xPropertiesWin->GetTotalLineCount(); m_xVertScroll->vadjustment_set_upper(nLineCount + 1); if (m_xPropertiesWin->GetHeight() < nLineCount * m_xPropertiesWin->GetLineHeight()) { m_xVertScroll->vadjustment_set_value(nLineCount + 1); ScrollHdl(*m_xVertScroll); } } void CustomPropertiesControl::SetCustomProperties(std::vector< std::unique_ptr >&& rProperties) { m_xPropertiesWin->SetCustomProperties(std::move(rProperties)); auto nLineCount = m_xPropertiesWin->GetTotalLineCount(); m_xVertScroll->vadjustment_set_upper(nLineCount + 1); } // class SfxCustomPropertiesPage ----------------------------------------- SfxCustomPropertiesPage::SfxCustomPropertiesPage(weld::Container* pPage, weld::DialogController* pController, const SfxItemSet& rItemSet ) : SfxTabPage(pPage, pController, u"sfx/ui/custominfopage.ui"_ustr, u"CustomInfoPage"_ustr, &rItemSet) , m_xPropertiesCtrl(new CustomPropertiesControl) , m_xAdd(m_xBuilder->weld_button(u"add"_ustr)) { m_xPropertiesCtrl->Init(*m_xBuilder); m_xAdd->connect_clicked(LINK(this, SfxCustomPropertiesPage, AddHdl)); } SfxCustomPropertiesPage::~SfxCustomPropertiesPage() { m_xPropertiesCtrl.reset(); } IMPL_LINK_NOARG(SfxCustomPropertiesPage, AddHdl, weld::Button&, void) { // tdf#115853: reload current lines before adding a brand new one // indeed the info are deleted by ClearCustomProperties // each time SfxDocumentInfoItem destructor is called SfxDocumentInfoItem pInfo; const Sequence< beans::PropertyValue > aPropertySeq = m_xPropertiesCtrl->GetCustomProperties(); for ( const auto& rProperty : aPropertySeq ) { if ( !rProperty.Name.isEmpty() ) { pInfo.AddCustomProperty( rProperty.Name, rProperty.Value ); } } Any aAny; m_xPropertiesCtrl->AddLine(aAny); } bool SfxCustomPropertiesPage::FillItemSet( SfxItemSet* rSet ) { const SfxDocumentInfoItem* pItem = nullptr; SfxDocumentInfoItem* pInfo = nullptr; bool bMustDelete = false; if (const SfxItemSet* pItemSet = GetDialogExampleSet()) { pItem = pItemSet->GetItemIfSet(SID_DOCINFO); if (!pItem) pInfo = const_cast(&rSet->Get( SID_DOCINFO )); else { bMustDelete = true; pInfo = new SfxDocumentInfoItem( *pItem ); } } if ( pInfo ) { // If it's a CMIS document, we can't save custom properties if ( pInfo->isCmisDocument( ) ) { if ( bMustDelete ) delete pInfo; return false; } pInfo->ClearCustomProperties(); const Sequence< beans::PropertyValue > aPropertySeq = m_xPropertiesCtrl->GetCustomProperties(); for ( const auto& rProperty : aPropertySeq ) { if ( !rProperty.Name.isEmpty() ) pInfo->AddCustomProperty( rProperty.Name, rProperty.Value ); } } if (pInfo) { rSet->Put(*pInfo); if ( bMustDelete ) delete pInfo; } return true; } void SfxCustomPropertiesPage::Reset( const SfxItemSet* rItemSet ) { m_xPropertiesCtrl->ClearAllLines(); const SfxDocumentInfoItem& rInfoItem = rItemSet->Get(SID_DOCINFO); std::vector< std::unique_ptr > aCustomProps = rInfoItem.GetCustomProperties(); // tdf#123919 - sort custom document properties auto const sort = comphelper::string::NaturalStringSorter( comphelper::getProcessComponentContext(), Application::GetSettings().GetLanguageTag().getLocale()); std::sort(aCustomProps.begin(), aCustomProps.end(), [&sort](const std::unique_ptr& rLHS, const std::unique_ptr& rRHS) { return sort.compare(rLHS->m_sName, rRHS->m_sName) < 0; }); m_xPropertiesCtrl->SetCustomProperties(std::move(aCustomProps)); } DeactivateRC SfxCustomPropertiesPage::DeactivatePage( SfxItemSet* /*pSet*/ ) { DeactivateRC nRet = DeactivateRC::LeavePage; if ( !m_xPropertiesCtrl->AreAllLinesValid() ) nRet = DeactivateRC::KeepPage; return nRet; } std::unique_ptr SfxCustomPropertiesPage::Create(weld::Container* pPage, weld::DialogController* pController, const SfxItemSet* rItemSet) { return std::make_unique(pPage, pController, *rItemSet); } CmisValue::CmisValue(weld::Widget* pParent, const OUString& aStr) : m_xBuilder(Application::CreateBuilder(pParent, u"sfx/ui/cmisline.ui"_ustr)) , m_xFrame(m_xBuilder->weld_frame(u"CmisFrame"_ustr)) , m_xValueEdit(m_xBuilder->weld_entry(u"value"_ustr)) { m_xValueEdit->show(); m_xValueEdit->set_text(aStr); } CmisDateTime::CmisDateTime(weld::Widget* pParent, const util::DateTime& aDateTime) : m_xBuilder(Application::CreateBuilder(pParent, u"sfx/ui/cmisline.ui"_ustr)) , m_xFrame(m_xBuilder->weld_frame(u"CmisFrame"_ustr)) , m_xDateField(new SvtCalendarBox(m_xBuilder->weld_menu_button(u"date"_ustr))) , m_xTimeField(m_xBuilder->weld_formatted_spin_button(u"time"_ustr)) , m_xFormatter(new weld::TimeFormatter(*m_xTimeField)) { m_xFormatter->SetExtFormat(ExtTimeFieldFormat::LongDuration); m_xFormatter->EnableEmptyField(false); m_xDateField->show(); m_xTimeField->show(); m_xDateField->set_date(Date(aDateTime)); m_xFormatter->SetTime(tools::Time(aDateTime)); } CmisYesNo::CmisYesNo(weld::Widget* pParent, bool bValue) : m_xBuilder(Application::CreateBuilder(pParent, u"sfx/ui/cmisline.ui"_ustr)) , m_xFrame(m_xBuilder->weld_frame(u"CmisFrame"_ustr)) , m_xYesButton(m_xBuilder->weld_radio_button(u"yes"_ustr)) , m_xNoButton(m_xBuilder->weld_radio_button(u"no"_ustr)) { m_xYesButton->show(); m_xNoButton->show(); if (bValue) m_xYesButton->set_active(true); else m_xNoButton->set_active(true); } // struct CmisPropertyLine --------------------------------------------- CmisPropertyLine::CmisPropertyLine(weld::Widget* pParent) : m_xBuilder(Application::CreateBuilder(pParent, u"sfx/ui/cmisline.ui"_ustr)) , m_sType(CMIS_TYPE_STRING) , m_bUpdatable(false) , m_bRequired(false) , m_bMultiValued(false) , m_bOpenChoice(false) , m_xFrame(m_xBuilder->weld_frame(u"CmisFrame"_ustr)) , m_xName(m_xBuilder->weld_label(u"name"_ustr)) , m_xType(m_xBuilder->weld_label(u"type"_ustr)) { m_xFrame->set_sensitive(true); } CmisPropertyLine::~CmisPropertyLine( ) { } // class CmisPropertiesWindow ----------------------------------------- CmisPropertiesWindow::CmisPropertiesWindow(std::unique_ptr xParent) : m_xBox(std::move(xParent)) , m_aNumberFormatter(::comphelper::getProcessComponentContext(), Application::GetSettings().GetLanguageTag().getLanguageType()) { } CmisPropertiesWindow::~CmisPropertiesWindow() { } void CmisPropertiesWindow::ClearAllLines() { m_aCmisPropertiesLines.clear(); } void CmisPropertiesWindow::AddLine( const OUString& sId, const OUString& sName, const OUString& sType, const bool bUpdatable, const bool bRequired, const bool bMultiValued, const bool bOpenChoice, Any& /*aChoices*/, Any const & rAny ) { std::unique_ptr pNewLine(new CmisPropertyLine(m_xBox.get())); pNewLine->m_sId = sId; pNewLine->m_sType = sType; pNewLine->m_bUpdatable = bUpdatable; pNewLine->m_bRequired = bRequired; pNewLine->m_bMultiValued = bMultiValued; pNewLine->m_bOpenChoice = bOpenChoice; if ( sType == CMIS_TYPE_INTEGER ) { Sequence< sal_Int64 > seqValue; rAny >>= seqValue; sal_uInt32 nIndex = m_aNumberFormatter.GetFormatIndex( NF_NUMBER_SYSTEM ); for (const auto& rValue : seqValue) { OUString sValue; m_aNumberFormatter.GetInputLineString( rValue, nIndex, sValue ); std::unique_ptr pValue(new CmisValue(m_xBox.get(), sValue)); pValue->m_xValueEdit->set_editable(bUpdatable); pNewLine->m_aValues.push_back( std::move(pValue) ); } } else if ( sType == CMIS_TYPE_DECIMAL ) { Sequence< double > seqValue; rAny >>= seqValue; sal_uInt32 nIndex = m_aNumberFormatter.GetFormatIndex( NF_NUMBER_SYSTEM ); for (const auto& rValue : seqValue) { OUString sValue; m_aNumberFormatter.GetInputLineString( rValue, nIndex, sValue ); std::unique_ptr pValue(new CmisValue(m_xBox.get(), sValue)); pValue->m_xValueEdit->set_editable(bUpdatable); pNewLine->m_aValues.push_back( std::move(pValue) ); } } else if ( sType == CMIS_TYPE_BOOL ) { Sequence seqValue; rAny >>= seqValue; for (const auto& rValue : seqValue) { std::unique_ptr pYesNo(new CmisYesNo(m_xBox.get(), rValue)); pYesNo->m_xYesButton->set_sensitive( bUpdatable ); pYesNo->m_xNoButton->set_sensitive( bUpdatable ); pNewLine->m_aYesNos.push_back( std::move(pYesNo) ); } } else if ( sType == CMIS_TYPE_STRING ) { Sequence< OUString > seqValue; rAny >>= seqValue; for (const auto& rValue : seqValue) { std::unique_ptr pValue(new CmisValue(m_xBox.get(), rValue)); pValue->m_xValueEdit->set_editable(bUpdatable); pNewLine->m_aValues.push_back( std::move(pValue) ); } } else if ( sType == CMIS_TYPE_DATETIME ) { Sequence< util::DateTime > seqValue; rAny >>= seqValue; for (const auto& rValue : seqValue) { std::unique_ptr pDateTime(new CmisDateTime(m_xBox.get(), rValue)); pDateTime->m_xDateField->set_sensitive(bUpdatable); pDateTime->m_xTimeField->set_sensitive(bUpdatable); pNewLine->m_aDateTimes.push_back( std::move(pDateTime) ); } } pNewLine->m_xName->set_label( sName ); pNewLine->m_xName->show(); pNewLine->m_xType->set_label( sType ); pNewLine->m_xType->show(); m_aCmisPropertiesLines.push_back( std::move(pNewLine) ); } Sequence< document::CmisProperty > CmisPropertiesWindow::GetCmisProperties() const { Sequence< document::CmisProperty > aPropertiesSeq( m_aCmisPropertiesLines.size() ); auto aPropertiesSeqRange = asNonConstRange(aPropertiesSeq); sal_Int32 i = 0; for ( auto& rxLine : m_aCmisPropertiesLines ) { CmisPropertyLine* pLine = rxLine.get(); aPropertiesSeqRange[i].Id = pLine->m_sId; aPropertiesSeqRange[i].Type = pLine->m_sType; aPropertiesSeqRange[i].Updatable = pLine->m_bUpdatable; aPropertiesSeqRange[i].Required = pLine->m_bRequired; aPropertiesSeqRange[i].OpenChoice = pLine->m_bOpenChoice; aPropertiesSeqRange[i].MultiValued = pLine->m_bMultiValued; OUString sPropertyName = pLine->m_xName->get_label(); if ( !sPropertyName.isEmpty() ) { aPropertiesSeqRange[i].Name = sPropertyName; OUString sType = pLine->m_xType->get_label(); if ( CMIS_TYPE_DECIMAL == sType ) { sal_uInt32 nIndex = const_cast< SvNumberFormatter& >( m_aNumberFormatter ).GetFormatIndex( NF_NUMBER_SYSTEM ); Sequence< double > seqValue( pLine->m_aValues.size( ) ); auto seqValueRange = asNonConstRange(seqValue); sal_Int32 k = 0; for ( const auto& rxValue : pLine->m_aValues ) { double dValue = 0.0; OUString sValue( rxValue->m_xValueEdit->get_text() ); bool bIsNum = const_cast< SvNumberFormatter& >( m_aNumberFormatter ). IsNumberFormat( sValue, nIndex, dValue ); if ( bIsNum ) seqValueRange[k] = dValue; ++k; } aPropertiesSeqRange[i].Value <<= seqValue; } else if ( CMIS_TYPE_INTEGER == sType ) { sal_uInt32 nIndex = const_cast< SvNumberFormatter& >( m_aNumberFormatter ).GetFormatIndex( NF_NUMBER_SYSTEM ); Sequence< sal_Int64 > seqValue( pLine->m_aValues.size( ) ); auto seqValueRange = asNonConstRange(seqValue); sal_Int32 k = 0; for ( const auto& rxValue : pLine->m_aValues ) { double dValue = 0; OUString sValue( rxValue->m_xValueEdit->get_text() ); bool bIsNum = const_cast< SvNumberFormatter& >( m_aNumberFormatter ). IsNumberFormat( sValue, nIndex, dValue ); if ( bIsNum ) seqValueRange[k] = static_cast(dValue); ++k; } aPropertiesSeqRange[i].Value <<= seqValue; } else if ( CMIS_TYPE_BOOL == sType ) { Sequence seqValue( pLine->m_aYesNos.size( ) ); sal_Bool* pseqValue = seqValue.getArray(); sal_Int32 k = 0; for ( const auto& rxYesNo : pLine->m_aYesNos ) { bool bValue = rxYesNo->m_xYesButton->get_active(); pseqValue[k] = bValue; ++k; } aPropertiesSeqRange[i].Value <<= seqValue; } else if ( CMIS_TYPE_DATETIME == sType ) { Sequence< util::DateTime > seqValue( pLine->m_aDateTimes.size( ) ); auto seqValueRange = asNonConstRange(seqValue); sal_Int32 k = 0; for ( const auto& rxDateTime : pLine->m_aDateTimes ) { Date aTmpDate = rxDateTime->m_xDateField->get_date(); tools::Time aTmpTime = rxDateTime->m_xFormatter->GetTime(); util::DateTime aDateTime( aTmpTime.GetNanoSec(), aTmpTime.GetSec(), aTmpTime.GetMin(), aTmpTime.GetHour(), aTmpDate.GetDay(), aTmpDate.GetMonth(), aTmpDate.GetYear(), true ); seqValueRange[k] = aDateTime; ++k; } aPropertiesSeqRange[i].Value <<= seqValue; } else { Sequence< OUString > seqValue( pLine->m_aValues.size( ) ); auto seqValueRange = asNonConstRange(seqValue); sal_Int32 k = 0; for ( const auto& rxValue : pLine->m_aValues ) { seqValueRange[k] = rxValue->m_xValueEdit->get_text(); ++k; } aPropertiesSeqRange[i].Value <<= seqValue; } } ++i; } return aPropertiesSeq; } CmisPropertiesControl::CmisPropertiesControl(weld::Builder& rBuilder) : m_aPropertiesWin(rBuilder.weld_container(u"CmisWindow"_ustr)) , m_xScrolledWindow(rBuilder.weld_scrolled_window(u"CmisScroll"_ustr)) { // set height to something small and force it to take the size // dictated by the other pages m_xScrolledWindow->set_size_request(-1, 42); } void CmisPropertiesControl::ClearAllLines() { m_aPropertiesWin.ClearAllLines(); } void CmisPropertiesControl::AddLine( const OUString& sId, const OUString& sName, const OUString& sType, const bool bUpdatable, const bool bRequired, const bool bMultiValued, const bool bOpenChoice, Any& aChoices, Any const & rAny ) { m_aPropertiesWin.AddLine( sId, sName, sType, bUpdatable, bRequired, bMultiValued, bOpenChoice, aChoices, rAny ); } // class SfxCmisPropertiesPage ----------------------------------------- SfxCmisPropertiesPage::SfxCmisPropertiesPage(weld::Container* pPage, weld::DialogController* pController, const SfxItemSet& rItemSet) : SfxTabPage(pPage, pController, u"sfx/ui/cmisinfopage.ui"_ustr, u"CmisInfoPage"_ustr, &rItemSet) , m_xPropertiesCtrl(new CmisPropertiesControl(*m_xBuilder)) { } SfxCmisPropertiesPage::~SfxCmisPropertiesPage() { m_xPropertiesCtrl.reset(); } bool SfxCmisPropertiesPage::FillItemSet( SfxItemSet* rSet ) { const SfxDocumentInfoItem* pItem = nullptr; SfxDocumentInfoItem* pInfo = nullptr; bool bMustDelete = false; if (const SfxItemSet* pItemSet = GetDialogExampleSet()) { pItem = pItemSet->GetItemIfSet(SID_DOCINFO); if (!pItem) pInfo = const_cast(&rSet->Get( SID_DOCINFO )); else { bMustDelete = true; pInfo = new SfxDocumentInfoItem( *pItem ); } } sal_Int32 modifiedNum = 0; if ( pInfo ) { Sequence< document::CmisProperty > aOldProps = pInfo->GetCmisProperties( ); Sequence< document::CmisProperty > aNewProps = m_xPropertiesCtrl->GetCmisProperties(); std::vector< document::CmisProperty > changedProps; for ( sal_Int32 i = 0; i< aNewProps.getLength( ); ++i ) { if ( aOldProps[i].Updatable && !aNewProps[i].Id.isEmpty( ) ) { if ( aOldProps[i].Type == CMIS_TYPE_DATETIME ) { Sequence< util::DateTime > oldValue; aOldProps[i].Value >>= oldValue; // We only edit hours and minutes // don't compare NanoSeconds and Seconds for ( auto& rDateTime : asNonConstRange(oldValue) ) { rDateTime.NanoSeconds = 0; rDateTime.Seconds = 0; } Sequence< util::DateTime > newValue; aNewProps[i].Value >>= newValue; if ( oldValue != newValue ) { modifiedNum++; changedProps.push_back( aNewProps[i] ); } } else if ( aOldProps[i].Value != aNewProps[i].Value ) { modifiedNum++; changedProps.push_back( aNewProps[i] ); } } } Sequence< document::CmisProperty> aModifiedProps( comphelper::containerToSequence(changedProps) ); pInfo->SetCmisProperties( aModifiedProps ); rSet->Put( *pInfo ); if ( bMustDelete ) delete pInfo; } return modifiedNum; } void SfxCmisPropertiesPage::Reset( const SfxItemSet* rItemSet ) { m_xPropertiesCtrl->ClearAllLines(); const SfxDocumentInfoItem& rInfoItem = rItemSet->Get(SID_DOCINFO); uno::Sequence< document::CmisProperty > aCmisProps = rInfoItem.GetCmisProperties(); for ( auto& rCmisProp : asNonConstRange(aCmisProps) ) { m_xPropertiesCtrl->AddLine(rCmisProp.Id, rCmisProp.Name, rCmisProp.Type, rCmisProp.Updatable, rCmisProp.Required, rCmisProp.MultiValued, rCmisProp.OpenChoice, rCmisProp.Choices, rCmisProp.Value); } } DeactivateRC SfxCmisPropertiesPage::DeactivatePage( SfxItemSet* /*pSet*/ ) { return DeactivateRC::LeavePage; } std::unique_ptr SfxCmisPropertiesPage::Create(weld::Container* pPage, weld::DialogController* pController, const SfxItemSet* rItemSet) { return std::make_unique(pPage, pController, *rItemSet); } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */