/* -*- 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 "SettingsTable.hxx" #include "TagLogger.hxx" #include #include #include #include #include #include #include #include #include #include #include #include #include "ConversionHelper.hxx" #include "DomainMapper.hxx" #include "util.hxx" using namespace com::sun::star; namespace writerfilter { namespace { /// Maps OOXML to SvxZoomType. sal_Int16 lcl_GetZoomType(Id nType) { switch (nType) { case NS_ooxml::LN_Value_doc_ST_Zoom_fullPage: return sal_Int16(SvxZoomType::WHOLEPAGE); case NS_ooxml::LN_Value_doc_ST_Zoom_bestFit: return sal_Int16(SvxZoomType::PAGEWIDTH); case NS_ooxml::LN_Value_doc_ST_Zoom_textFit: return sal_Int16(SvxZoomType::OPTIMAL); } return sal_Int16(SvxZoomType::PERCENT); } } namespace dmapper { namespace { /** Document protection restrictions * * This element specifies the set of document protection restrictions which have been applied to the contents of a * WordprocessingML document.These restrictions should be enforced by applications editing this document * when the enforcement attribute is turned on, and ignored(but persisted) otherwise.Document protection is a * set of restrictions used to prevent unintentional changes to all or part of a WordprocessingML document. */ struct DocumentProtection_Impl { /** Document Editing Restrictions * * Possible values: * - NS_ooxml::LN_Value_doc_ST_DocProtect_none * - NS_ooxml::LN_Value_doc_ST_DocProtect_readOnly * - NS_ooxml::LN_Value_doc_ST_DocProtect_comments * - NS_ooxml::LN_Value_doc_ST_DocProtect_trackedChanges * - NS_ooxml::LN_Value_doc_ST_DocProtect_forms */ sal_Int32 m_nEdit; bool m_bEnforcement; bool m_bFormatting; /** Provider type * * Possible values: * "rsaAES" - NS_ooxml::LN_Value_doc_ST_CryptProv_rsaAES * "rsaFull" - NS_ooxml::LN_Value_doc_ST_CryptProv_rsaFull */ sal_Int32 m_nCryptProviderType; OUString m_sCryptAlgorithmClass; OUString m_sCryptAlgorithmType; OUString m_sCryptAlgorithmSid; sal_Int32 m_CryptSpinCount; OUString m_sHash; OUString m_sSalt; DocumentProtection_Impl() : m_nEdit(NS_ooxml::LN_Value_doc_ST_DocProtect_none) // Specifies that no editing restrictions have been applied to the document , m_bEnforcement(false) , m_bFormatting(false) , m_nCryptProviderType(NS_ooxml::LN_Value_doc_ST_CryptProv_rsaAES) , m_sCryptAlgorithmClass("hash") , m_sCryptAlgorithmType("typeAny") , m_CryptSpinCount(0) { } css::uno::Sequence toSequence() const; bool enabled() const { return ! isNone(); } bool isNone() const { return m_nEdit == NS_ooxml::LN_Value_doc_ST_DocProtect_none; }; }; } css::uno::Sequence DocumentProtection_Impl::toSequence() const { std::vector documentProtection; if (enabled()) { // w:edit { beans::PropertyValue aValue; aValue.Name = "edit"; switch (m_nEdit) { case NS_ooxml::LN_Value_doc_ST_DocProtect_none: aValue.Value <<= OUString("none"); break; case NS_ooxml::LN_Value_doc_ST_DocProtect_readOnly: aValue.Value <<= OUString("readOnly"); break; case NS_ooxml::LN_Value_doc_ST_DocProtect_comments: aValue.Value <<= OUString("comments"); break; case NS_ooxml::LN_Value_doc_ST_DocProtect_trackedChanges: aValue.Value <<= OUString("trackedChanges"); break; case NS_ooxml::LN_Value_doc_ST_DocProtect_forms: aValue.Value <<= OUString("forms"); break; default: { #ifdef DBG_UTIL TagLogger::getInstance().element("unhandled"); #endif } } documentProtection.push_back(aValue); } // w:enforcement if (m_bEnforcement) { beans::PropertyValue aValue; aValue.Name = "enforcement"; aValue.Value <<= OUString("1"); documentProtection.push_back(aValue); } // w:formatting if (m_bFormatting) { beans::PropertyValue aValue; aValue.Name = "formatting"; aValue.Value <<= OUString("1"); documentProtection.push_back(aValue); } // w:cryptProviderType { beans::PropertyValue aValue; aValue.Name = "cryptProviderType"; if (m_nCryptProviderType == NS_ooxml::LN_Value_doc_ST_CryptProv_rsaAES) aValue.Value <<= OUString("rsaAES"); else if (m_nCryptProviderType == NS_ooxml::LN_Value_doc_ST_CryptProv_rsaFull) aValue.Value <<= OUString("rsaFull"); documentProtection.push_back(aValue); } // w:cryptAlgorithmClass { beans::PropertyValue aValue; aValue.Name = "cryptAlgorithmClass"; aValue.Value <<= m_sCryptAlgorithmClass; documentProtection.push_back(aValue); } // w:cryptAlgorithmType { beans::PropertyValue aValue; aValue.Name = "cryptAlgorithmType"; aValue.Value <<= m_sCryptAlgorithmType; documentProtection.push_back(aValue); } // w:cryptAlgorithmSid { beans::PropertyValue aValue; aValue.Name = "cryptAlgorithmSid"; aValue.Value <<= m_sCryptAlgorithmSid; documentProtection.push_back(aValue); } // w:cryptSpinCount { beans::PropertyValue aValue; aValue.Name = "cryptSpinCount"; aValue.Value <<= OUString::number(m_CryptSpinCount); documentProtection.push_back(aValue); } // w:hash { beans::PropertyValue aValue; aValue.Name = "hash"; aValue.Value <<= m_sHash; documentProtection.push_back(aValue); } // w:salt { beans::PropertyValue aValue; aValue.Name = "salt"; aValue.Value <<= m_sSalt; documentProtection.push_back(aValue); } } return comphelper::containerToSequence(documentProtection); } struct SettingsTable_Impl { int m_nDefaultTabStop; bool m_bRecordChanges; bool m_bShowInsDelChanges; bool m_bShowFormattingChanges; bool m_bShowMarkupChanges; bool m_bLinkStyles; sal_Int16 m_nZoomFactor; sal_Int16 m_nZoomType = 0; sal_Int32 m_nWordCompatibilityMode; Id m_nView; bool m_bEvenAndOddHeaders; bool m_bUsePrinterMetrics; bool embedTrueTypeFonts; bool embedSystemFonts; bool m_bDoNotUseHTMLParagraphAutoSpacing; bool m_bNoColumnBalance; bool m_bAutoHyphenation; bool m_bNoHyphenateCaps; sal_Int16 m_nHyphenationZone; bool m_bWidowControl; bool m_bLongerSpaceSequence; bool m_bSplitPgBreakAndParaMark; bool m_bMirrorMargin; bool m_bDoNotExpandShiftReturn; bool m_bProtectForm; bool m_bRedlineProtection; OUString m_sRedlineProtectionKey; bool m_bReadOnly; bool m_bDisplayBackgroundShape; bool m_bNoLeading = false; OUString m_sDecimalSymbol; OUString m_sListSeparator; uno::Sequence m_pThemeFontLangProps; std::vector m_aCompatSettings; uno::Sequence m_pCurrentCompatSetting; OUString m_sCurrentDatabaseDataSource; DocumentProtection_Impl m_DocumentProtection; SettingsTable_Impl() : m_nDefaultTabStop( 720 ) //default is 1/2 in , m_bRecordChanges(false) , m_bShowInsDelChanges(true) , m_bShowFormattingChanges(false) , m_bShowMarkupChanges(true) , m_bLinkStyles(false) , m_nZoomFactor(0) , m_nWordCompatibilityMode(-1) , m_nView(0) , m_bEvenAndOddHeaders(false) , m_bUsePrinterMetrics(false) , embedTrueTypeFonts(false) , embedSystemFonts(false) , m_bDoNotUseHTMLParagraphAutoSpacing(false) , m_bNoColumnBalance(false) , m_bAutoHyphenation(false) , m_bNoHyphenateCaps(false) , m_nHyphenationZone(0) , m_bWidowControl(false) , m_bLongerSpaceSequence(false) , m_bSplitPgBreakAndParaMark(false) , m_bMirrorMargin(false) , m_bDoNotExpandShiftReturn(false) , m_bProtectForm(false) , m_bRedlineProtection(false) , m_sRedlineProtectionKey() , m_bReadOnly(false) , m_bDisplayBackgroundShape(false) , m_sDecimalSymbol(".") , m_sListSeparator(",") , m_pThemeFontLangProps(3) , m_pCurrentCompatSetting(3) {} }; SettingsTable::SettingsTable(const DomainMapper& rDomainMapper) : LoggedProperties("SettingsTable") , LoggedTable("SettingsTable") , m_pImpl( new SettingsTable_Impl ) { if (rDomainMapper.IsRTFImport()) { // HTML paragraph auto-spacing is opt-in for RTF, opt-out for OOXML. m_pImpl->m_bDoNotUseHTMLParagraphAutoSpacing = true; // Longer space sequence is opt-in for RTF, and not in OOXML. m_pImpl->m_bLongerSpaceSequence = true; } } SettingsTable::~SettingsTable() { } void SettingsTable::lcl_attribute(Id nName, Value & val) { int nIntValue = val.getInt(); OUString sStringValue = val.getString(); switch(nName) { case NS_ooxml::LN_CT_Zoom_percent: m_pImpl->m_nZoomFactor = nIntValue; break; case NS_ooxml::LN_CT_Zoom_val: m_pImpl->m_nZoomType = lcl_GetZoomType(nIntValue); break; case NS_ooxml::LN_CT_Language_val: m_pImpl->m_pThemeFontLangProps[0].Name = "val"; m_pImpl->m_pThemeFontLangProps[0].Value <<= sStringValue; break; case NS_ooxml::LN_CT_Language_eastAsia: m_pImpl->m_pThemeFontLangProps[1].Name = "eastAsia"; m_pImpl->m_pThemeFontLangProps[1].Value <<= sStringValue; break; case NS_ooxml::LN_CT_Language_bidi: m_pImpl->m_pThemeFontLangProps[2].Name = "bidi"; m_pImpl->m_pThemeFontLangProps[2].Value <<= sStringValue; break; case NS_ooxml::LN_CT_View_val: m_pImpl->m_nView = nIntValue; break; case NS_ooxml::LN_CT_CompatSetting_name: m_pImpl->m_pCurrentCompatSetting[0].Name = "name"; m_pImpl->m_pCurrentCompatSetting[0].Value <<= sStringValue; break; case NS_ooxml::LN_CT_CompatSetting_uri: m_pImpl->m_pCurrentCompatSetting[1].Name = "uri"; m_pImpl->m_pCurrentCompatSetting[1].Value <<= sStringValue; break; case NS_ooxml::LN_CT_CompatSetting_val: m_pImpl->m_pCurrentCompatSetting[2].Name = "val"; m_pImpl->m_pCurrentCompatSetting[2].Value <<= sStringValue; break; case NS_ooxml::LN_CT_DocProtect_edit: // 92037 m_pImpl->m_DocumentProtection.m_nEdit = nIntValue; // multiple DocProtect_edits should not exist. If they do, last one wins m_pImpl->m_bRedlineProtection = false; m_pImpl->m_bProtectForm = false; m_pImpl->m_bReadOnly = false; switch (nIntValue) { case NS_ooxml::LN_Value_doc_ST_DocProtect_trackedChanges: { m_pImpl->m_bRedlineProtection = true; m_pImpl->m_sRedlineProtectionKey = m_pImpl->m_DocumentProtection.m_sHash; break; } case NS_ooxml::LN_Value_doc_ST_DocProtect_forms: m_pImpl->m_bProtectForm = true; break; case NS_ooxml::LN_Value_doc_ST_DocProtect_readOnly: m_pImpl->m_bReadOnly = true; break; } break; case NS_ooxml::LN_CT_DocProtect_enforcement: // 92039 m_pImpl->m_DocumentProtection.m_bEnforcement = (nIntValue != 0); break; case NS_ooxml::LN_CT_DocProtect_formatting: // 92038 m_pImpl->m_DocumentProtection.m_bFormatting = (nIntValue != 0); break; case NS_ooxml::LN_AG_Password_cryptProviderType: // 92025 m_pImpl->m_DocumentProtection.m_nCryptProviderType = nIntValue; break; case NS_ooxml::LN_AG_Password_cryptAlgorithmClass: // 92026 if (nIntValue == NS_ooxml::LN_Value_doc_ST_AlgClass_hash) // 92023 m_pImpl->m_DocumentProtection.m_sCryptAlgorithmClass = "hash"; break; case NS_ooxml::LN_AG_Password_cryptAlgorithmType: // 92027 if (nIntValue == NS_ooxml::LN_Value_doc_ST_AlgType_typeAny) // 92024 m_pImpl->m_DocumentProtection.m_sCryptAlgorithmType = "typeAny"; break; case NS_ooxml::LN_AG_Password_cryptAlgorithmSid: // 92028 m_pImpl->m_DocumentProtection.m_sCryptAlgorithmSid = sStringValue; break; case NS_ooxml::LN_AG_Password_cryptSpinCount: // 92029 m_pImpl->m_DocumentProtection.m_CryptSpinCount = nIntValue; break; case NS_ooxml::LN_AG_Password_hash: // 92035 m_pImpl->m_DocumentProtection.m_sHash = sStringValue; break; case NS_ooxml::LN_AG_Password_salt: // 92036 m_pImpl->m_DocumentProtection.m_sSalt = sStringValue; break; case NS_ooxml::LN_CT_TrackChangesView_insDel: m_pImpl->m_bShowInsDelChanges = (nIntValue != 0); break; case NS_ooxml::LN_CT_TrackChangesView_formatting: m_pImpl->m_bShowFormattingChanges = (nIntValue != 0); break; case NS_ooxml::LN_CT_TrackChangesView_markup: m_pImpl->m_bShowMarkupChanges = (nIntValue != 0); break; default: { #ifdef DBG_UTIL TagLogger::getInstance().element("unhandled"); #endif } } } void SettingsTable::lcl_sprm(Sprm& rSprm) { sal_uInt32 nSprmId = rSprm.getId(); Value::Pointer_t pValue = rSprm.getValue(); sal_Int32 nIntValue = pValue->getInt(); switch(nSprmId) { case NS_ooxml::LN_CT_Settings_zoom: // 92469; case NS_ooxml::LN_CT_Settings_proofState: // 92489; case NS_ooxml::LN_CT_Settings_attachedTemplate: // 92491; case NS_ooxml::LN_CT_Settings_hdrShapeDefaults: // 92544; case NS_ooxml::LN_CT_Settings_footnotePr: // 92545; case NS_ooxml::LN_CT_Settings_endnotePr: // 92546; case NS_ooxml::LN_CT_Settings_compat: // 92547; case NS_ooxml::LN_CT_Settings_themeFontLang: // 92552; case NS_ooxml::LN_CT_Settings_shapeDefaults: // 92560; case NS_ooxml::LN_CT_Settings_view: //PropertySetValues - need to be resolved { resolveSprmProps(*this, rSprm); } break; case NS_ooxml::LN_CT_Settings_stylePaneFormatFilter: // 92493; break; case NS_ooxml::LN_CT_Settings_defaultTabStop: // 92505; m_pImpl->m_nDefaultTabStop = nIntValue; break; case NS_ooxml::LN_CT_Settings_linkStyles: // 92663; m_pImpl->m_bLinkStyles = nIntValue; break; case NS_ooxml::LN_CT_Settings_evenAndOddHeaders: m_pImpl->m_bEvenAndOddHeaders = nIntValue; break; case NS_ooxml::LN_CT_Settings_noPunctuationKerning: // 92526; break; case NS_ooxml::LN_CT_Settings_characterSpacingControl: // 92527; // doNotCompress, compressPunctuation, compressPunctuationAndJapaneseKana break; case NS_ooxml::LN_CT_Settings_doNotIncludeSubdocsInStats: // 92554; // Do Not Include Content in Text Boxes, Footnotes, and Endnotes in Document Statistics) break; case NS_ooxml::LN_CT_Settings_decimalSymbol: // 92562; m_pImpl->m_sDecimalSymbol = pValue->getString(); break; case NS_ooxml::LN_CT_Settings_listSeparator: // 92563; m_pImpl->m_sListSeparator = pValue->getString(); break; case NS_ooxml::LN_CT_Settings_rsids: // 92549; revision save Ids - probably not necessary break; case NS_ooxml::LN_CT_Settings_hyphenationZone: // 92508; m_pImpl->m_nHyphenationZone = nIntValue; break; case NS_ooxml::LN_CT_Compat_useFELayout: // 92422; // useFELayout (Do Not Bypass East Asian/Complex Script Layout Code - support of old versions of Word - ignored) break; case NS_ooxml::LN_CT_Settings_trackRevisions: { m_pImpl->m_bRecordChanges = bool(rSprm.getValue( )->getInt( ) ); } break; case NS_ooxml::LN_CT_Settings_revisionView: resolveSprmProps(*this, rSprm); break; case NS_ooxml::LN_CT_Settings_documentProtection: resolveSprmProps(*this, rSprm); break; case NS_ooxml::LN_CT_Compat_usePrinterMetrics: m_pImpl->m_bUsePrinterMetrics = nIntValue; break; case NS_ooxml::LN_CT_Settings_embedTrueTypeFonts: m_pImpl->embedTrueTypeFonts = nIntValue != 0; break; case NS_ooxml::LN_CT_Settings_embedSystemFonts: m_pImpl->embedSystemFonts = nIntValue != 0; break; case NS_ooxml::LN_CT_Compat_doNotUseHTMLParagraphAutoSpacing: m_pImpl->m_bDoNotUseHTMLParagraphAutoSpacing = nIntValue; break; case NS_ooxml::LN_CT_Compat_splitPgBreakAndParaMark: m_pImpl->m_bSplitPgBreakAndParaMark = nIntValue; break; case NS_ooxml::LN_CT_Settings_mirrorMargins: m_pImpl->m_bMirrorMargin = nIntValue; break; case NS_ooxml::LN_CT_Settings_mailMerge: { writerfilter::Reference::Pointer_t pProperties = rSprm.getProps(); if (pProperties) pProperties->resolve(*this); } break; case NS_ooxml::LN_CT_MailMerge_query: { // try to get the "database.table" name from the query saved previously OUString sVal = pValue->getString(); if ( sVal.endsWith("$") && sVal.indexOf(".dbo.") > 0 ) { sal_Int32 nSpace = sVal.lastIndexOf(' '); sal_Int32 nDbo = sVal.lastIndexOf(".dbo."); if ( nSpace > 0 && nSpace < nDbo - 1 ) { m_pImpl->m_sCurrentDatabaseDataSource = OUString::Concat(sVal.subView(nSpace + 1, nDbo - nSpace - 1)) + sVal.subView(nDbo + 4, sVal.getLength() - nDbo - 5); } } } break; case NS_ooxml::LN_CT_Compat_compatSetting: { writerfilter::Reference::Pointer_t pProperties = rSprm.getProps(); if (pProperties) { pProperties->resolve(*this); beans::PropertyValue aValue; aValue.Name = "compatSetting"; aValue.Value <<= m_pImpl->m_pCurrentCompatSetting; m_pImpl->m_aCompatSettings.push_back(aValue); } } break; case NS_ooxml::LN_CT_Compat_noColumnBalance: m_pImpl->m_bNoColumnBalance = nIntValue; break; case NS_ooxml::LN_CT_Settings_autoHyphenation: m_pImpl->m_bAutoHyphenation = nIntValue; break; case NS_ooxml::LN_CT_Settings_doNotHyphenateCaps: m_pImpl->m_bNoHyphenateCaps = nIntValue; break; case NS_ooxml::LN_CT_Settings_widowControl: m_pImpl->m_bWidowControl = nIntValue; break; case NS_ooxml::LN_CT_Settings_longerSpaceSequence: m_pImpl->m_bLongerSpaceSequence = nIntValue; break; case NS_ooxml::LN_CT_Compat_doNotExpandShiftReturn: m_pImpl->m_bDoNotExpandShiftReturn = true; break; case NS_ooxml::LN_CT_Settings_displayBackgroundShape: m_pImpl->m_bDisplayBackgroundShape = nIntValue; break; case NS_ooxml::LN_CT_Compat_noLeading: m_pImpl->m_bNoLeading = nIntValue != 0; break; default: { #ifdef DBG_UTIL TagLogger::getInstance().element("unhandled"); #endif } } } void SettingsTable::lcl_entry(writerfilter::Reference::Pointer_t ref) { ref->resolve(*this); } //returns default TabStop in 1/100th mm int SettingsTable::GetDefaultTabStop() const { return ConversionHelper::convertTwipToMM100( m_pImpl->m_nDefaultTabStop ); } bool SettingsTable::GetLinkStyles() const { return m_pImpl->m_bLinkStyles; } sal_Int16 SettingsTable::GetZoomFactor() const { return m_pImpl->m_nZoomFactor; } sal_Int16 SettingsTable::GetZoomType() const { return m_pImpl->m_nZoomType; } Id SettingsTable::GetView() const { return m_pImpl->m_nView; } bool SettingsTable::GetUsePrinterMetrics() const { return m_pImpl->m_bUsePrinterMetrics; } bool SettingsTable::GetEvenAndOddHeaders() const { return m_pImpl->m_bEvenAndOddHeaders; } bool SettingsTable::GetEmbedTrueTypeFonts() const { return m_pImpl->embedTrueTypeFonts; } bool SettingsTable::GetEmbedSystemFonts() const { return m_pImpl->embedSystemFonts; } bool SettingsTable::GetDoNotUseHTMLParagraphAutoSpacing() const { return m_pImpl->m_bDoNotUseHTMLParagraphAutoSpacing; } bool SettingsTable::GetNoColumnBalance() const { return m_pImpl->m_bNoColumnBalance; } bool SettingsTable::GetSplitPgBreakAndParaMark() const { return m_pImpl->m_bSplitPgBreakAndParaMark; } bool SettingsTable::GetMirrorMarginSettings() const { return m_pImpl->m_bMirrorMargin; } bool SettingsTable::GetDisplayBackgroundShape() const { return m_pImpl->m_bDisplayBackgroundShape; } bool SettingsTable::GetDoNotExpandShiftReturn() const { return m_pImpl->m_bDoNotExpandShiftReturn; } bool SettingsTable::GetProtectForm() const { return m_pImpl->m_bProtectForm && m_pImpl->m_DocumentProtection.m_bEnforcement; } bool SettingsTable::GetReadOnly() const { return m_pImpl->m_bReadOnly && m_pImpl->m_DocumentProtection.m_bEnforcement; } bool SettingsTable::GetNoHyphenateCaps() const { return m_pImpl->m_bNoHyphenateCaps; } sal_Int16 SettingsTable::GetHyphenationZone() const { return m_pImpl->m_nHyphenationZone; } OUString SettingsTable::GetDecimalSymbol() const { return m_pImpl->m_sDecimalSymbol; } OUString SettingsTable::GetListSeparator() const { return m_pImpl->m_sListSeparator; } uno::Sequence const & SettingsTable::GetThemeFontLangProperties() const { return m_pImpl->m_pThemeFontLangProps; } uno::Sequence SettingsTable::GetCompatSettings() const { if ( GetWordCompatibilityMode() == -1 ) { // the default value for an undefined compatibilityMode is 12 (Word 2007) uno::Sequence aCompatSetting( comphelper::InitPropertySequence({ { "name", uno::Any(OUString("compatibilityMode")) }, { "uri", uno::Any(OUString("http://schemas.microsoft.com/office/word")) }, { "val", uno::Any(OUString("12")) } //12: Use word processing features specified in ECMA-376. This is the default. })); beans::PropertyValue aValue; aValue.Name = "compatSetting"; aValue.Value <<= aCompatSetting; m_pImpl->m_aCompatSettings.push_back(aValue); } return comphelper::containerToSequence(m_pImpl->m_aCompatSettings); } css::uno::Sequence SettingsTable::GetDocumentProtectionSettings() const { return m_pImpl->m_DocumentProtection.toSequence(); } const OUString & SettingsTable::GetCurrentDatabaseDataSource() const { return m_pImpl->m_sCurrentDatabaseDataSource; } static bool lcl_isDefault(const uno::Reference& xPropertyState, const OUString& rPropertyName) { return xPropertyState->getPropertyState(rPropertyName) == beans::PropertyState_DEFAULT_VALUE; } void SettingsTable::ApplyProperties(uno::Reference const& xDoc) { uno::Reference< beans::XPropertySet> xDocProps( xDoc, uno::UNO_QUERY ); if (GetWordCompatibilityMode() <= 14) { uno::Reference xTextFactory(xDoc, uno::UNO_QUERY_THROW); uno::Reference xDocumentSettings(xTextFactory->createInstance("com.sun.star.document.Settings"), uno::UNO_QUERY_THROW); xDocumentSettings->setPropertyValue("MsWordCompMinLineHeightByFly", uno::makeAny(true)); } // Show changes value if (xDocProps.is()) { bool bHideChanges = !m_pImpl->m_bShowInsDelChanges || !m_pImpl->m_bShowMarkupChanges; xDocProps->setPropertyValue("ShowChanges", uno::makeAny( !bHideChanges || m_pImpl->m_bShowFormattingChanges ) ); } // Record changes value if (xDocProps.is()) { xDocProps->setPropertyValue("RecordChanges", uno::makeAny( m_pImpl->m_bRecordChanges ) ); // Password protected Record changes if ( m_pImpl->m_bRecordChanges && m_pImpl->m_bRedlineProtection && m_pImpl->m_DocumentProtection.m_bEnforcement ) { // use dummy protection key to forbid disabling of Record changes without a notice // (extending the recent GrabBag support) TODO support password verification... css::uno::Sequence aDummyKey(1); aDummyKey[0] = 1; xDocProps->setPropertyValue("RedlineProtectionKey", uno::makeAny( aDummyKey )); } } // Auto hyphenation: turns on hyphenation by default, may still disable it at a paragraph level. // Situation is similar for RTF_WIDOWCTRL, which turns on widow / orphan control by default. if (!(m_pImpl->m_bAutoHyphenation || m_pImpl->m_bNoHyphenateCaps || m_pImpl->m_bWidowControl)) return; uno::Reference xStyleFamiliesSupplier(xDoc, uno::UNO_QUERY); if (!xStyleFamiliesSupplier.is()) return; uno::Reference xStyleFamilies = xStyleFamiliesSupplier->getStyleFamilies(); uno::Reference xParagraphStyles = xStyleFamilies->getByName("ParagraphStyles").get< uno::Reference >(); uno::Reference xDefault = xParagraphStyles->getByName("Standard").get< uno::Reference >(); uno::Reference xPropertyState(xDefault, uno::UNO_QUERY); if (m_pImpl->m_bAutoHyphenation && lcl_isDefault(xPropertyState, "ParaIsHyphenation")) { uno::Reference xPropertySet(xDefault, uno::UNO_QUERY); xPropertySet->setPropertyValue("ParaIsHyphenation", uno::makeAny(true)); } if (m_pImpl->m_bNoHyphenateCaps) { uno::Reference xPropertySet(xDefault, uno::UNO_QUERY); xPropertySet->setPropertyValue("ParaHyphenationNoCaps", uno::makeAny(true)); } if (m_pImpl->m_bWidowControl && lcl_isDefault(xPropertyState, "ParaWidows") && lcl_isDefault(xPropertyState, "ParaOrphans")) { uno::Reference xPropertySet(xDefault, uno::UNO_QUERY); uno::Any aAny = uno::makeAny(static_cast(2)); xPropertySet->setPropertyValue("ParaWidows", aAny); xPropertySet->setPropertyValue("ParaOrphans", aAny); } } bool SettingsTable::GetCompatSettingValue( std::u16string_view sCompatName ) const { bool bRet = false; for (const auto& rProp : m_pImpl->m_aCompatSettings) { if (rProp.Name == "compatSetting") //always true { css::uno::Sequence aCurrentCompatSettings; rProp.Value >>= aCurrentCompatSettings; OUString sName; aCurrentCompatSettings[0].Value >>= sName; if ( sName != sCompatName ) continue; OUString sUri; aCurrentCompatSettings[1].Value >>= sUri; if ( sUri != "http://schemas.microsoft.com/office/word" ) continue; OUString sVal; aCurrentCompatSettings[2].Value >>= sVal; // if repeated, what happens? Last one wins bRet = sVal.toBoolean(); } } return bRet; } //Keep this function in-sync with the one in sw/.../docxattributeoutput.cxx sal_Int32 SettingsTable::GetWordCompatibilityMode() const { if ( m_pImpl->m_nWordCompatibilityMode != -1 ) return m_pImpl->m_nWordCompatibilityMode; for (const auto& rProp : m_pImpl->m_aCompatSettings) { if (rProp.Name == "compatSetting") //always true { css::uno::Sequence aCurrentCompatSettings; rProp.Value >>= aCurrentCompatSettings; OUString sName; aCurrentCompatSettings[0].Value >>= sName; if ( sName != "compatibilityMode" ) continue; OUString sUri; aCurrentCompatSettings[1].Value >>= sUri; if ( sUri != "http://schemas.microsoft.com/office/word" ) continue; OUString sVal; aCurrentCompatSettings[2].Value >>= sVal; const sal_Int32 nValidMode = sVal.toInt32(); // if repeated, highest mode wins in MS Word. 11 is the first valid mode. if ( nValidMode > 10 && nValidMode > m_pImpl->m_nWordCompatibilityMode ) m_pImpl->m_nWordCompatibilityMode = nValidMode; } } return m_pImpl->m_nWordCompatibilityMode; } bool SettingsTable::GetLongerSpaceSequence() const { return m_pImpl->m_bLongerSpaceSequence; } bool SettingsTable::GetNoLeading() const { return m_pImpl->m_bNoLeading; } }//namespace dmapper } //namespace writerfilter /* vim:set shiftwidth=4 softtabstop=4 expandtab: */