/* -*- 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 "rtl/textcvt.h" #include "xlat.hxx" #include namespace { #define MAX_CVT_SELECT 6 class ConverterCache { public: explicit ConverterCache(); ~ConverterCache(); sal_uInt16 convertOne( int nSelect, sal_Unicode ); void convertStr( int nSelect, const sal_Unicode* pSrc, sal_uInt16* pDst, int nCount ); protected: void ensureConverter( int nSelect ); private: rtl_UnicodeToTextConverter maConverterCache[ MAX_CVT_SELECT+1 ]; rtl_UnicodeToTextContext maContexts[ MAX_CVT_SELECT+1 ]; }; ConverterCache::ConverterCache() { for( int i = 0; i <= MAX_CVT_SELECT; ++i) { maConverterCache[i] = NULL; maContexts[i] = NULL; } } ConverterCache::~ConverterCache() { for( int i = 0; i <= MAX_CVT_SELECT; ++i) { if( !maContexts[i] ) continue; rtl_destroyUnicodeToTextContext( maConverterCache[i], maContexts[i] ); rtl_destroyUnicodeToTextConverter( maConverterCache[i] ); } } void ConverterCache::ensureConverter( int nSelect ) { // DBG_ASSERT( (2<=nSelect) && (nSelect<=MAX_CVT_SELECT)), "invalid XLAT.Converter requested" ); rtl_UnicodeToTextContext aContext = maContexts[ nSelect ]; if( !aContext ) { rtl_TextEncoding eRecodeFrom = RTL_TEXTENCODING_UNICODE; switch( nSelect ) { default: nSelect = 1; // fall through to unicode recoding case 1: eRecodeFrom = RTL_TEXTENCODING_UNICODE; break; case 2: eRecodeFrom = RTL_TEXTENCODING_SHIFT_JIS; break; case 3: eRecodeFrom = RTL_TEXTENCODING_GB_2312; break; case 4: eRecodeFrom = RTL_TEXTENCODING_BIG5; break; case 5: eRecodeFrom = RTL_TEXTENCODING_MS_949; break; case 6: eRecodeFrom = RTL_TEXTENCODING_MS_1361; break; } rtl_UnicodeToTextConverter aRecodeConverter = rtl_createUnicodeToTextConverter( eRecodeFrom ); maConverterCache[ nSelect ] = aRecodeConverter; aContext = rtl_createUnicodeToTextContext( aRecodeConverter ); maContexts[ nSelect ] = aContext; } rtl_resetUnicodeToTextContext( maConverterCache[ nSelect ], aContext ); } sal_uInt16 ConverterCache::convertOne( int nSelect, sal_Unicode aChar ) { ensureConverter( nSelect ); sal_Unicode aUCS2Char = aChar; sal_Char aTempArray[8]; sal_Size nTempSize; sal_uInt32 nCvtInfo; // TODO: use direct unicode->mbcs converter should there ever be one int nCodeLen = rtl_convertUnicodeToText( maConverterCache[ nSelect ], maContexts[ nSelect ], &aUCS2Char, 1, aTempArray, sizeof(aTempArray), RTL_UNICODETOTEXT_FLAGS_UNDEFINED_0 | RTL_UNICODETOTEXT_FLAGS_INVALID_0, &nCvtInfo, &nTempSize ); sal_uInt16 aCode = aTempArray[0]; for( int i = 1; i < nCodeLen; ++i ) aCode = (aCode << 8) + (aTempArray[i] & 0xFF); return aCode; } void ConverterCache::convertStr( int nSelect, const sal_Unicode* pSrc, sal_uInt16* pDst, int nCount ) { ensureConverter( nSelect ); for( int n = 0; n < nCount; ++n ) { sal_Unicode aUCS2Char = pSrc[n]; sal_Char aTempArray[8]; sal_Size nTempSize; sal_uInt32 nCvtInfo; // assume that non-unicode-fonts do not support codepoints >U+FFFF // TODO: use direct unicode->mbcs converter should there ever be one int nCodeLen = rtl_convertUnicodeToText( maConverterCache[ nSelect ], maContexts[ nSelect ], &aUCS2Char, 1, aTempArray, sizeof(aTempArray), RTL_UNICODETOTEXT_FLAGS_UNDEFINED_0 | RTL_UNICODETOTEXT_FLAGS_INVALID_0, &nCvtInfo, &nTempSize ); sal_uInt16 aCode = aTempArray[0]; for( int i = 1; i < nCodeLen; ++i ) aCode = (aCode << 8) + (aTempArray[i] & 0xFF); pDst[n] = aCode; } } } // anonymous namespace namespace vcl { static ConverterCache aCC; sal_uInt16 TranslateChar12(sal_uInt16 src) { return aCC.convertOne( 2, src); } sal_uInt16 TranslateChar13(sal_uInt16 src) { return aCC.convertOne( 3, src); } sal_uInt16 TranslateChar14(sal_uInt16 src) { return aCC.convertOne( 4, src); } sal_uInt16 TranslateChar15(sal_uInt16 src) { return aCC.convertOne( 5, src); } sal_uInt16 TranslateChar16(sal_uInt16 src) { return aCC.convertOne( 6, src); } void TranslateString12(sal_uInt16 *src, sal_uInt16 *dst, sal_uInt32 n) { aCC.convertStr( 2, src, dst, n); } void TranslateString13(sal_uInt16 *src, sal_uInt16 *dst, sal_uInt32 n) { aCC.convertStr( 3, src, dst, n); } void TranslateString14(sal_uInt16 *src, sal_uInt16 *dst, sal_uInt32 n) { aCC.convertStr( 4, src, dst, n); } void TranslateString15(sal_uInt16 *src, sal_uInt16 *dst, sal_uInt32 n) { aCC.convertStr( 5, src, dst, n); } void TranslateString16(sal_uInt16 *src, sal_uInt16 *dst, sal_uInt32 n) { aCC.convertStr( 6, src, dst, n); } } // namespace vcl /* vim:set shiftwidth=4 softtabstop=4 expandtab: */