/* -*- 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 using namespace utl; using namespace com::sun::star; using namespace com::sun::star::uno; using namespace com::sun::star::beans; const char cReplacement[] = "Replacement"; const char cFontPairs[] = "FontPairs"; const char cReplaceFont[] = "ReplaceFont"; const char cSubstituteFont[]= "SubstituteFont"; const char cOnScreenOnly[] = "OnScreenOnly"; const char cAlways[] = "Always"; typedef std::vector SubstitutionStructArr; struct SvtFontSubstConfig_Impl { SubstitutionStructArr aSubstArr; }; SvtFontSubstConfig::SvtFontSubstConfig() : ConfigItem("Office.Common/Font/Substitution"), bIsEnabled(false), pImpl(new SvtFontSubstConfig_Impl) { Sequence aNames { cReplacement }; Sequence aValues = GetProperties(aNames); DBG_ASSERT(aValues.getConstArray()[0].hasValue(), "no value available"); if(aValues.getConstArray()[0].hasValue()) bIsEnabled = *o3tl::doAccess(aValues.getConstArray()[0]); OUString sPropPrefix(cFontPairs); const Sequence aNodeNames = GetNodeNames(sPropPrefix, ConfigNameFormat::LocalPath); Sequence aPropNames(aNodeNames.getLength() * 4); OUString* pNames = aPropNames.getArray(); sal_Int32 nName = 0; sPropPrefix += "/"; for(const OUString& rNodeName : aNodeNames) { OUString sStart = sPropPrefix + rNodeName + "/"; pNames[nName++] = sStart + cReplaceFont; pNames[nName++] = sStart + cSubstituteFont; pNames[nName++] = sStart + cAlways; pNames[nName++] = sStart + cOnScreenOnly; } Sequence aNodeValues = GetProperties(aPropNames); const Any* pNodeValues = aNodeValues.getConstArray(); nName = 0; for(sal_Int32 nNode = 0; nNode < aNodeNames.getLength(); nNode++) { SubstitutionStruct aInsert; pNodeValues[nName++] >>= aInsert.sFont; pNodeValues[nName++] >>= aInsert.sReplaceBy; aInsert.bReplaceAlways = *o3tl::doAccess(pNodeValues[nName++]); aInsert.bReplaceOnScreenOnly = *o3tl::doAccess(pNodeValues[nName++]); pImpl->aSubstArr.push_back(aInsert); } } SvtFontSubstConfig::~SvtFontSubstConfig() { } void SvtFontSubstConfig::Notify( const css::uno::Sequence< OUString >& ) { } void SvtFontSubstConfig::ImplCommit() { PutProperties({cReplacement}, {css::uno::Any(bIsEnabled)}); OUString sNode(cFontPairs); if(pImpl->aSubstArr.empty()) ClearNodeSet(sNode); else { Sequence aSetValues(4 * pImpl->aSubstArr.size()); PropertyValue* pSetValues = aSetValues.getArray(); sal_Int32 nSetValue = 0; const OUString sReplaceFont(cReplaceFont); const OUString sSubstituteFont(cSubstituteFont); const OUString sAlways(cAlways); const OUString sOnScreenOnly(cOnScreenOnly); for(size_t i = 0; i < pImpl->aSubstArr.size(); i++) { OUString sPrefix = sNode + "/_" + OUString::number(i) + "/"; SubstitutionStruct& rSubst = pImpl->aSubstArr[i]; pSetValues[nSetValue].Name = sPrefix; pSetValues[nSetValue].Name += sReplaceFont; pSetValues[nSetValue++].Value <<= rSubst.sFont; pSetValues[nSetValue].Name = sPrefix; pSetValues[nSetValue].Name += sSubstituteFont; pSetValues[nSetValue++].Value <<= rSubst.sReplaceBy; pSetValues[nSetValue].Name = sPrefix; pSetValues[nSetValue].Name += sAlways; pSetValues[nSetValue++].Value <<= rSubst.bReplaceAlways; pSetValues[nSetValue].Name = sPrefix; pSetValues[nSetValue].Name += sOnScreenOnly; pSetValues[nSetValue++].Value <<= rSubst.bReplaceOnScreenOnly; } ReplaceSetProperties(sNode, aSetValues); } } sal_Int32 SvtFontSubstConfig::SubstitutionCount() const { return pImpl->aSubstArr.size(); } void SvtFontSubstConfig::ClearSubstitutions() { pImpl->aSubstArr.clear(); } const SubstitutionStruct* SvtFontSubstConfig::GetSubstitution(sal_Int32 nPos) { sal_Int32 nCount = static_cast(pImpl->aSubstArr.size()); DBG_ASSERT(nPos >= 0 && nPos < nCount, "illegal array index"); if(nPos >= 0 && nPos < nCount) return &pImpl->aSubstArr[nPos]; return nullptr; } void SvtFontSubstConfig::AddSubstitution(const SubstitutionStruct& rToAdd) { pImpl->aSubstArr.push_back(rToAdd); } void SvtFontSubstConfig::Apply() { OutputDevice::BeginFontSubstitution(); // remove old substitutions OutputDevice::RemoveFontsSubstitute(); // read new substitutions sal_Int32 nCount = IsEnabled() ? SubstitutionCount() : 0; for (sal_Int32 i = 0; i < nCount; i++) { AddFontSubstituteFlags nFlags = AddFontSubstituteFlags::NONE; const SubstitutionStruct* pSubs = GetSubstitution(i); if(pSubs->bReplaceAlways) nFlags |= AddFontSubstituteFlags::ALWAYS; if(pSubs->bReplaceOnScreenOnly) nFlags |= AddFontSubstituteFlags::ScreenOnly; OutputDevice::AddFontSubstitute( pSubs->sFont, pSubs->sReplaceBy, nFlags ); } OutputDevice::EndFontSubstitution(); } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */