/* -*- 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 RscString::RscString( Atom nId, RESOURCE_TYPE nTypeId ) : RscTop( nId, nTypeId ) , pRefClass(nullptr) { } ERRTYPE RscString::SetString( const RSCINST & rInst, const char * pStr ) { char * pTmp; ERRTYPE aError; if( aError.IsOk() ) { reinterpret_cast(rInst.pData)->bDflt = false; pTmp = reinterpret_cast(rInst.pData)->pStr; if( pTmp ) { rtl_freeMemory( pTmp ); pTmp = nullptr; } if( pStr ) { sal_uInt32 nLen = strlen( pStr ) +1; pTmp = static_cast(rtl_allocateMemory( nLen )); memcpy( pTmp, pStr, nLen ); } reinterpret_cast(rInst.pData)->pStr = pTmp; } return aError; } ERRTYPE RscString::GetString( const RSCINST & rInst, char ** ppStr ) { *ppStr = reinterpret_cast(rInst.pData)->pStr; return ERR_OK; } ERRTYPE RscString::GetRef( const RSCINST & rInst, RscId * pRscId ) { *pRscId = reinterpret_cast(rInst.pData)->aRefId; return ERR_OK; } ERRTYPE RscString::SetRef( const RSCINST & rInst, const RscId & rRefId ) { if( pRefClass ) { reinterpret_cast(rInst.pData)->aRefId = rRefId; reinterpret_cast(rInst.pData)->bDflt = false; } else return ERR_REFNOTALLOWED; return ERR_OK; } RSCINST RscString::Create( RSCINST * pInst, const RSCINST & rDflt, bool bOwnClass ) { RSCINST aInst; if( !pInst ) { aInst.pClass = this; aInst.pData = static_cast( rtl_allocateMemory( sizeof( RscStringInst ) )); } else aInst = *pInst; if( !bOwnClass && rDflt.IsInst() ) bOwnClass = rDflt.pClass->InHierarchy( this ); reinterpret_cast(aInst.pData)->aRefId.Create(); reinterpret_cast(aInst.pData)->pStr = nullptr; reinterpret_cast(aInst.pData)->bDflt = true; if( bOwnClass ) { reinterpret_cast(aInst.pData)->aRefId = reinterpret_cast(rDflt.pData)->aRefId; SetString( aInst, reinterpret_cast(rDflt.pData)->pStr ); reinterpret_cast(aInst.pData)->bDflt = reinterpret_cast(rDflt.pData)->bDflt; } return aInst; } void RscString::Destroy( const RSCINST & rInst ) { if( reinterpret_cast(rInst.pData)->pStr ) rtl_freeMemory( reinterpret_cast(rInst.pData)->pStr ); reinterpret_cast(rInst.pData)->aRefId.Destroy(); } bool RscString::IsValueDefault( const RSCINST & rInst, CLASS_DATA pDef ) { RscStringInst * pData = reinterpret_cast(rInst.pData); RscStringInst * pDefData = reinterpret_cast(pDef); if( pDef ) { if( pData->aRefId.IsId() || pDefData->aRefId.IsId() ) { if( pData->aRefId.aExp.IsNumber() && pDefData->aRefId.aExp.IsNumber() ) { // check whether reference identifiers are equal if( pData->aRefId.GetNumber() == pDefData->aRefId.GetNumber() ) { return true; } } } else { bool bStrEmpty = false; bool bDefStrEmpty = false; if( pData->pStr ) { bStrEmpty = ('\0' == *pData->pStr); } else bStrEmpty = true; if( pDefData->pStr ) { bDefStrEmpty = ('\0' == *pDefData->pStr); } else bDefStrEmpty = true; if( !bStrEmpty || !bDefStrEmpty ) { return false; } return true; } } return false; } void RscString::WriteSrc( const RSCINST & rInst, FILE * fOutput, RscTypCont *, sal_uInt32, const char * ) { if ( reinterpret_cast(rInst.pData)->aRefId.IsId() ) { fprintf( fOutput, "%s", reinterpret_cast(rInst.pData)->aRefId.GetName().getStr() ); } else { RscStringInst * pStrI = reinterpret_cast(rInst.pData); if( pStrI->pStr ){ //char * pChangeTab = RscChar::GetChangeTab(); sal_uInt32 n = 0; sal_uInt32 nPos, nSlashPos; do { fputc( '\"', fOutput ); nSlashPos = nPos = 0; while( pStrI->pStr[ n ] && (nPos < 72 || nPos - nSlashPos <= 3) ) { // after \ drop at least 3 characters \xa7 fputc( pStrI->pStr[ n ], fOutput ); if( pStrI->pStr[ n ] == '\\' ) nSlashPos = nPos; n++; nPos++; } fputc( '\"', fOutput ); if( pStrI->pStr[ n ] ) // end not reached yet { fputc( '\n', fOutput ); } } while( pStrI->pStr[ n ] ); } else fprintf( fOutput, "\"\"" ); } } ERRTYPE RscString::WriteRc( const RSCINST & rInst, RscWriteRc & rMem, RscTypCont * pTC, sal_uInt32 nDeep ) { ERRTYPE aError; ObjNode * pObjNode = nullptr; if( reinterpret_cast(rInst.pData)->aRefId.IsId() ) { RscId aId( reinterpret_cast(rInst.pData)->aRefId ); RSCINST aTmpI; aTmpI.pClass = pRefClass; while( aError.IsOk() && aId.IsId() ) { // increment and test to avoid endless recursion nDeep++; if( nDeep > nRefDeep ) aError = ERR_REFTODEEP; else { pObjNode = pRefClass->GetObjNode( aId ); if( pObjNode ) { aTmpI.pData = pObjNode->GetRscObj(); aError = pRefClass->GetRef( aTmpI, &aId ); } else { if( pTC ) { OStringBuffer aMsg(pHS->getString( pRefClass->GetId())); aMsg.append(' ').append(aId.GetName()); aError = WRN_STR_REFNOTFOUND; pTC->pEH->Error( aError, rInst.pClass, RscId(), aMsg.getStr() ); } break; } } } } if( aError.IsOk() ) { if( pObjNode ) { RSCINST aRefI; aRefI = RSCINST( pRefClass, pObjNode->GetRscObj() ); aError = aRefI.pClass->WriteRc( aRefI, rMem, pTC, nDeep ); } else { if( reinterpret_cast(rInst.pData)->pStr && pTC ) { char * pStr = RscChar::MakeUTF8( reinterpret_cast(rInst.pData)->pStr, pTC->GetSourceCharSet() ); rMem.PutUTF8( pStr ); rtl_freeMemory( pStr ); } else rMem.PutUTF8( reinterpret_cast(rInst.pData)->pStr ); } } return aError; } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */