/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* * This file is part of the LibreOffice project. * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ #include #include #include #include #include #include #include #include #include #include #include #include namespace { // Extract strings from nodes on all level recursively void lcl_ExtractLevel( const xmlDocPtr pSource, const xmlNodePtr pRoot, const xmlChar* pNodeName, PoOfstream& rPOStream ) { if( !pRoot->children ) { return; } for( xmlNodePtr pCurrent = pRoot->children->next; pCurrent; pCurrent = pCurrent->next) { if (!xmlStrcmp(pCurrent->name, pNodeName)) { xmlChar* pID = xmlGetProp(pCurrent, reinterpret_cast("id")); xmlChar* pText = xmlGetProp(pCurrent, reinterpret_cast("title")); common::writePoEntry( "Treex"_ostr, rPOStream, pSource->name, helper::xmlStrToOString( pNodeName ), helper::xmlStrToOString( pID ), OString(), OString(), helper::xmlStrToOString( pText )); xmlFree( pID ); xmlFree( pText ); lcl_ExtractLevel( pSource, pCurrent, reinterpret_cast("node"), rPOStream ); } } } // Update id and content of the topic xmlNodePtr lcl_UpdateTopic( const xmlNodePtr pCurrent, std::string_view rXhpRoot ) { xmlNodePtr pReturn = pCurrent; xmlChar* pID = xmlGetProp(pReturn, reinterpret_cast("id")); const OString sID = helper::xmlStrToOString( pID ); xmlFree( pID ); const std::string_view::size_type nFirstSlash = sID.indexOf('/'); const auto nAfterSlash = (nFirstSlash != std::string_view::npos) ? (nFirstSlash + 1) : 0; // Update id attribute of topic { std::u16string_view::size_type nXhpSlash = rXhpRoot.rfind('/'); const auto nAfterXhpSlash = (nXhpSlash != std::u16string_view::npos) ? (nXhpSlash + 1) : 0; OString sNewID = OString::Concat(sID.subView( 0, nAfterSlash )) + rXhpRoot.substr( nAfterXhpSlash ) + sID.subView( sID.indexOf( '/', nAfterSlash ) ); xmlSetProp( pReturn, reinterpret_cast("id"), reinterpret_cast(sNewID.getStr())); } const OString sXhpPath = OString::Concat(rXhpRoot) + sID.subView(sID.indexOf('/', nAfterSlash)); xmlDocPtr pXhpFile = xmlParseFile( sXhpPath.getStr() ); // if xhpfile is missing than put this topic into comment if ( !pXhpFile ) { xmlNodePtr pTemp = pReturn; xmlChar* sNewID = xmlGetProp(pReturn, reinterpret_cast("id")); xmlChar* sComment = xmlStrcat( xmlCharStrdup("removed "), sNewID ); pReturn = xmlNewComment( sComment ); xmlReplaceNode( pTemp, pReturn ); xmlFree( pTemp ); xmlFree( sNewID ); xmlFree( sComment ); } // update topic's content on the basis of xhpfile's title else { xmlNodePtr pXhpNode = xmlDocGetRootElement( pXhpFile ); for( pXhpNode = pXhpNode->children; pXhpNode; pXhpNode = pXhpNode->children ) { while( pXhpNode->type != XML_ELEMENT_NODE ) { pXhpNode = pXhpNode->next; } if(!xmlStrcmp(pXhpNode->name, reinterpret_cast("title"))) { xmlChar* sTitle = xmlNodeListGetString(pXhpFile, pXhpNode->children, 1); OString sNewTitle = helper::xmlStrToOString( sTitle ). replaceAll("$[officename]"_ostr,"%PRODUCTNAME"_ostr). replaceAll("$[officeversion]"_ostr,"%PRODUCTVERSION"_ostr); xmlChar *xmlString = xmlEncodeSpecialChars(nullptr, reinterpret_cast( sNewTitle.getStr() )); xmlNodeSetContent( pReturn, xmlString); xmlFree( xmlString ); xmlFree( sTitle ); break; } } if( !pXhpNode ) { std::cerr << "Treex error: Cannot find title in " << sXhpPath << std::endl; pReturn = nullptr; } xmlFreeDoc( pXhpFile ); xmlCleanupParser(); } return pReturn; } // Localize title attribute of help_section and node tags void lcl_MergeLevel( xmlDocPtr io_pSource, const xmlNodePtr pRoot, const xmlChar * pNodeName, MergeDataFile* pMergeDataFile, const OString& rLang, const OString& rXhpRoot ) { if( !pRoot->children ) { return; } for( xmlNodePtr pCurrent = pRoot->children; pCurrent; pCurrent = pCurrent->next) { if( !xmlStrcmp(pCurrent->name, pNodeName) ) { if( rLang != "en-US" ) { OString sNewText; xmlChar* pID = xmlGetProp(pCurrent, reinterpret_cast("id")); ResData aResData( helper::xmlStrToOString( pID ), static_cast(io_pSource->name) ); xmlFree( pID ); aResData.sResTyp = helper::xmlStrToOString( pNodeName ); if( pMergeDataFile ) { MergeEntrys* pEntrys = pMergeDataFile->GetMergeEntrys( &aResData ); if( pEntrys ) { pEntrys->GetText( sNewText, rLang ); } } else if( rLang == "qtz" ) { xmlChar* pText = xmlGetProp(pCurrent, reinterpret_cast("title")); const OString sOriginText = helper::xmlStrToOString(pText); xmlFree( pText ); sNewText = MergeEntrys::GetQTZText(aResData, sOriginText); } if( !sNewText.isEmpty() ) { xmlSetProp( pCurrent, reinterpret_cast("title"), reinterpret_cast(sNewText.getStr())); } } lcl_MergeLevel( io_pSource, pCurrent, reinterpret_cast("node"), pMergeDataFile, rLang, rXhpRoot ); } else if( !xmlStrcmp(pCurrent->name, reinterpret_cast("topic")) ) { pCurrent = lcl_UpdateTopic( pCurrent, rXhpRoot ); } } } } TreeParser::TreeParser( const OString& rInputFile, OString _sLang ) : m_pSource( nullptr ) , m_sLang(std::move( _sLang )) , m_bIsInitialized( false ) { m_pSource = xmlParseFile( rInputFile.getStr() ); if ( !m_pSource ) { std::cerr << "Treex error: Cannot open source file: " << rInputFile << std::endl; return; } if( !m_pSource->name ) { m_pSource->name = static_cast(xmlMalloc(strlen(rInputFile.getStr())+1)); strcpy( m_pSource->name, rInputFile.getStr() ); } m_bIsInitialized = true; } TreeParser::~TreeParser() { // be sure m_pSource is freed if (m_bIsInitialized) xmlFreeDoc( m_pSource ); } void TreeParser::Extract( const OString& rPOFile ) { assert( m_bIsInitialized ); PoOfstream aPOStream( rPOFile, PoOfstream::APP ); if( !aPOStream.isOpen() ) { std::cerr << "Treex error: Cannot open po file for extract: " << rPOFile << std::endl; return; } xmlNodePtr pRootNode = xmlDocGetRootElement( m_pSource ); lcl_ExtractLevel( m_pSource, pRootNode, reinterpret_cast("help_section"), aPOStream ); xmlFreeDoc( m_pSource ); xmlCleanupParser(); aPOStream.close(); m_bIsInitialized = false; } void TreeParser::Merge( const OString &rMergeSrc, const OString &rDestinationFile, const OString &rXhpRoot ) { assert( m_bIsInitialized ); const xmlNodePtr pRootNode = xmlDocGetRootElement( m_pSource ); std::unique_ptr pMergeDataFile; if( m_sLang != "qtz" && m_sLang != "en-US" ) { pMergeDataFile.reset(new MergeDataFile( rMergeSrc, static_cast( m_pSource->name ), false, false )); const std::vector vLanguages = pMergeDataFile->GetLanguages(); if( !vLanguages.empty() && vLanguages[0] != m_sLang ) { std::cerr << ("Treex error: given language conflicts with language of" " Mergedata file: ") << m_sLang << " - " << vLanguages[0] << std::endl; return; } } lcl_MergeLevel( m_pSource, pRootNode, reinterpret_cast("help_section"), pMergeDataFile.get(), m_sLang, rXhpRoot ); pMergeDataFile.reset(); xmlSaveFile( rDestinationFile.getStr(), m_pSource ); xmlFreeDoc( m_pSource ); xmlCleanupParser(); m_bIsInitialized = false; } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */