/* -*- 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 #include #include #include #include #include using namespace com::sun::star::uno; using namespace com::sun::star::i18n; using namespace com::sun::star::lang; namespace i18npool { #define TmItem1( name ) \ {TransliterationModules_##name, TransliterationModulesNew_##name, #name} #define TmItem2( name ) \ {TransliterationModules(0), TransliterationModulesNew_##name, #name} namespace { // Ignore Module list struct TMList { TransliterationModules tm; TransliterationModulesNew tmn; const char *implName; }; } TMList const TMlist[] = { // Modules ModulesNew TmItem1 (IGNORE_CASE), // 0. (1<<8 256) (7) TmItem1 (IGNORE_WIDTH), // 1. (1<<9 512) (8) TmItem1 (IGNORE_KANA), // 2. (1<<10 1024) (9) // No enum define for this trans. application has to use impl name to load it // TmItem1 (IGNORE_CASE_SIMPLE), // (1<<11 1024) (66) TmItem1 (IgnoreTraditionalKanji_ja_JP), // 3. (1<<12 4096) (10) TmItem1 (IgnoreTraditionalKana_ja_JP), // 4. (1<<13 8192) (11) TmItem1 (IgnoreMinusSign_ja_JP), // 5. (1<<13 16384) (12) TmItem1 (IgnoreIterationMark_ja_JP), // 6. (1<<14 32768) (13) TmItem1 (IgnoreSeparator_ja_JP), // 7. (1<<15 65536) (14) TmItem1 (IgnoreSize_ja_JP), // 15. (1<<23 16777216) (22) TmItem1 (IgnoreMiddleDot_ja_JP), // 17. (1<<25 67108864) (24) TmItem1 (IgnoreSpace_ja_JP), // 18. (1<<26 134217728) (25) TmItem1 (IgnoreZiZu_ja_JP), // 8. (1<<16 131072) (15) TmItem1 (IgnoreBaFa_ja_JP), // 9. (1<<17 262144) (16) TmItem1 (IgnoreTiJi_ja_JP), // 10. (1<<18 524288) (17) TmItem1 (IgnoreHyuByu_ja_JP), // 11. (1<<19 1048576) (18) TmItem1 (IgnoreSeZe_ja_JP), // 12. (1<<20 2097152) (19) TmItem1 (IgnoreIandEfollowedByYa_ja_JP), // 13. (1<<21 4194304) (20) TmItem1 (IgnoreKiKuFollowedBySa_ja_JP), // 14. (1<<22 8388608) (21) TmItem1 (IgnoreProlongedSoundMark_ja_JP), // 16. (1<<24 33554432) (23) TmItem1 (UPPERCASE_LOWERCASE), // 19. (1) (1) TmItem1 (LOWERCASE_UPPERCASE), // 20. (2) (2) TmItem1 (HALFWIDTH_FULLWIDTH), // 21. (3) (3) TmItem1 (FULLWIDTH_HALFWIDTH), // 22. (4) (4) TmItem1 (KATAKANA_HIRAGANA), // 23. (5) (5) TmItem1 (HIRAGANA_KATAKANA), // 24. (6) (6) TmItem1 (SmallToLarge_ja_JP), // 25. (1<<27 268435456) (26) TmItem1 (LargeToSmall_ja_JP), // 26. (1<<28 536870912) (27) TmItem2 (NumToTextLower_zh_CN), // 27. () (28) TmItem2 (NumToTextUpper_zh_CN), // 28. () (29) TmItem2 (NumToTextLower_zh_TW), // 29. () (30) TmItem2 (NumToTextUpper_zh_TW), // 30. () (31) TmItem2 (NumToTextFormalHangul_ko), // 31. () (32) TmItem2 (NumToTextFormalLower_ko), // 32. () (33) TmItem2 (NumToTextFormalUpper_ko), // 33. () (34) TmItem2 (NumToTextInformalHangul_ko), // 34. () (35) TmItem2 (NumToTextInformalLower_ko), // 35. () (36) TmItem2 (NumToTextInformalUpper_ko), // 36. () (37) TmItem2 (NumToCharLower_zh_CN), // 37. () (38) TmItem2 (NumToCharUpper_zh_CN), // 38. () (39) TmItem2 (NumToCharLower_zh_TW), // 39. () (40) TmItem2 (NumToCharUpper_zh_TW), // 40. () (41) TmItem2 (NumToCharHangul_ko), // 41. () (42) TmItem2 (NumToCharLower_ko), // 42. () (43) TmItem2 (NumToCharUpper_ko), // 43. () (44) TmItem2 (NumToCharFullwidth), // 44. () (45) TmItem2 (NumToCharKanjiShort_ja_JP), // 45. () (46) TmItem2 (TextToNumLower_zh_CN), // 46. () (47) TmItem2 (TextToNumUpper_zh_CN), // 47. () (48) TmItem2 (TextToNumLower_zh_TW), // 48. () (49) TmItem2 (TextToNumUpper_zh_TW), // 49. () (50) TmItem2 (TextToNumFormalHangul_ko), // 50. () (51) TmItem2 (TextToNumFormalLower_ko), // 51. () (52) TmItem2 (TextToNumFormalUpper_ko), // 52. () (53) TmItem2 (TextToNumInformalHangul_ko), // 53. () (54) TmItem2 (TextToNumInformalLower_ko), // 54. () (55) TmItem2 (TextToNumInformalUpper_ko), // 55. () (56) TmItem2 (CharToNumLower_zh_CN), // 56. () (59) TmItem2 (CharToNumUpper_zh_CN), // 57. () (60) TmItem2 (CharToNumLower_zh_TW), // 58. () (61) TmItem2 (CharToNumUpper_zh_TW), // 59. () (62) TmItem2 (CharToNumHangul_ko), // 60. () (63) TmItem2 (CharToNumLower_ko), // 61. () (64) TmItem2 (CharToNumUpper_ko), // 62. () (65) // no enum defined for these trans. application has to use impl name to load them // TmItem2 (NumToCharArabic_Indic), // () (67) // TmItem2 (NumToCharEstern_Arabic_Indic),// () (68) // TmItem2 (NumToCharIndic), // () (69) // TmItem2 (NumToCharThai), // () (70) {TransliterationModules(0), TransliterationModulesNew(0), nullptr} }; // Constructor/Destructor TransliterationImpl::TransliterationImpl(const Reference & xContext) : mxContext(xContext) { numCascade = 0; caseignoreOnly = true; mxLocaledata.set(LocaleData2::create(xContext)); } TransliterationImpl::~TransliterationImpl() { mxLocaledata.clear(); clear(); } // Methods OUString SAL_CALL TransliterationImpl::getName() { if (numCascade == 1 && bodyCascade[0].is()) return bodyCascade[0]->getName(); if (numCascade < 1) return ( OUString("Not Loaded")); throw RuntimeException(); } sal_Int16 SAL_CALL TransliterationImpl::getType() { if (numCascade > 1) return (TransliterationType::CASCADE|TransliterationType::IGNORE); if (numCascade > 0 && bodyCascade[0].is()) return bodyCascade[0]->getType(); throw RuntimeException(); } static TransliterationModules operator&(TransliterationModules lhs, TransliterationModules rhs) { return TransliterationModules(sal_Int32(lhs) & sal_Int32(rhs)); } static TransliterationModules operator|(TransliterationModules lhs, TransliterationModules rhs) { return TransliterationModules(sal_Int32(lhs) | sal_Int32(rhs)); } void SAL_CALL TransliterationImpl::loadModule( TransliterationModules modType, const Locale& rLocale ) { clear(); if (bool(modType & TransliterationModules_IGNORE_MASK) && bool(modType & TransliterationModules_NON_IGNORE_MASK)) { throw RuntimeException(); } else if (bool(modType & TransliterationModules_IGNORE_MASK)) { #define TransliterationModules_IGNORE_CASE_MASK (TransliterationModules_IGNORE_CASE | \ TransliterationModules_IGNORE_WIDTH | \ TransliterationModules_IGNORE_KANA) TransliterationModules mask = ((modType & TransliterationModules_IGNORE_CASE_MASK) == modType) ? TransliterationModules_IGNORE_CASE_MASK : TransliterationModules_IGNORE_MASK; for (sal_Int16 i = 0; bool(TMlist[i].tm & mask); i++) { if (bool(modType & TMlist[i].tm)) if (loadModuleByName(OUString::createFromAscii(TMlist[i].implName), bodyCascade[numCascade], rLocale)) numCascade++; } // additional transliterations from TransliterationModulesExtra (we cannot extend TransliterationModules) if (bool(modType & TransliterationModules(TransliterationModulesExtra::IGNORE_DIACRITICS_CTL))) { if (loadModuleByName(u"ignoreDiacritics_CTL", bodyCascade[numCascade], rLocale)) numCascade++; } if (bool(modType & TransliterationModules(TransliterationModulesExtra::IGNORE_KASHIDA_CTL))) if (loadModuleByName(u"ignoreKashida_CTL", bodyCascade[numCascade], rLocale)) numCascade++; } else if (bool(modType & TransliterationModules_NON_IGNORE_MASK)) { for (sal_Int16 i = 0; bool(TMlist[i].tm); i++) { if (TMlist[i].tm == modType) { if (loadModuleByName(OUString::createFromAscii(TMlist[i].implName), bodyCascade[numCascade], rLocale)) numCascade++; break; } } } } void SAL_CALL TransliterationImpl::loadModuleNew( const Sequence < TransliterationModulesNew > & modType, const Locale& rLocale ) { clear(); TransliterationModules mask = TransliterationModules_END_OF_MODULE; sal_Int32 count = modType.getLength(); if (count > maxCascade) throw RuntimeException(); // could not handle more than maxCascade for (sal_Int32 i = 0; i < count; i++) { for (sal_Int16 j = 0; bool(TMlist[j].tmn); j++) { if (TMlist[j].tmn == modType[i]) { if (mask == TransliterationModules_END_OF_MODULE) mask = bool(TMlist[i].tm) && bool(TMlist[i].tm & TransliterationModules_IGNORE_MASK) ? TransliterationModules_IGNORE_MASK : TransliterationModules_NON_IGNORE_MASK; else if (mask == TransliterationModules_IGNORE_MASK && (TMlist[i].tm&TransliterationModules_IGNORE_MASK) == TransliterationModules_END_OF_MODULE) throw RuntimeException(); // could not mess up ignore trans. with non_ignore trans. if (loadModuleByName(OUString::createFromAscii(TMlist[j].implName), bodyCascade[numCascade], rLocale)) numCascade++; break; } } } } void SAL_CALL TransliterationImpl::loadModuleByImplName(const OUString& implName, const Locale& rLocale) { clear(); if (loadModuleByName(implName, bodyCascade[numCascade], rLocale)) numCascade++; } void SAL_CALL TransliterationImpl::loadModulesByImplNames(const Sequence< OUString >& implNameList, const Locale& rLocale ) { if (implNameList.getLength() > maxCascade || implNameList.getLength() <= 0) throw RuntimeException(); clear(); for (const auto& rName : implNameList) if (loadModuleByName(rName, bodyCascade[numCascade], rLocale)) numCascade++; } Sequence SAL_CALL TransliterationImpl::getAvailableModules( const Locale& rLocale, sal_Int16 sType ) { const Sequence &translist = mxLocaledata->getTransliterations(rLocale); std::vector r; r.reserve(translist.getLength()); Reference body; for (const auto& rTrans : translist) { if (loadModuleByName(rTrans, body, rLocale)) { if (body->getType() & sType) r.push_back(rTrans); body.clear(); } } return comphelper::containerToSequence(r); } OUString SAL_CALL TransliterationImpl::transliterate( const OUString& inStr, sal_Int32 startPos, sal_Int32 nCount, Sequence< sal_Int32 >& offset ) { if (numCascade == 0) return inStr; if (numCascade == 1) { if ( startPos == 0 && nCount == inStr.getLength() ) return bodyCascade[0]->transliterate( inStr, 0, nCount, offset); else { OUString tmpStr = inStr.copy(startPos, nCount); tmpStr = bodyCascade[0]->transliterate(tmpStr, 0, nCount, offset); if ( startPos ) { for (sal_Int32 & j : asNonConstRange(offset)) j += startPos; } return tmpStr; } } else { OUString tmpStr = inStr.copy(startPos, nCount); auto [begin, end] = asNonConstRange(offset); std::iota(begin, end, startPos); Sequence from(nCount); Sequence to = offset; for (sal_Int32 i = 0; i < numCascade; i++) { tmpStr = bodyCascade[i]->transliterate(tmpStr, 0, nCount, from); nCount = tmpStr.getLength(); assert(from.getLength() == nCount); from.swap(to); for (sal_Int32& ix : asNonConstRange(to)) ix = from[ix]; } offset = to; return tmpStr; } } OUString SAL_CALL TransliterationImpl::folding( const OUString& inStr, sal_Int32 startPos, sal_Int32 nCount, Sequence< sal_Int32 >& offset ) { if (numCascade == 0) return inStr; if (offset.getLength() != nCount) offset.realloc(nCount); if (numCascade == 1) { if ( startPos == 0 && nCount == inStr.getLength() ) return bodyCascade[0]->folding( inStr, 0, nCount, offset); else { OUString tmpStr = inStr.copy(startPos, nCount); tmpStr = bodyCascade[0]->folding(tmpStr, 0, nCount, offset); if ( startPos ) { for (sal_Int32 & j : asNonConstRange(offset)) j += startPos; } return tmpStr; } } else { OUString tmpStr = inStr.copy(startPos, nCount); auto [begin, end] = asNonConstRange(offset); std::iota(begin, end, startPos); Sequence from; Sequence to = offset; for (sal_Int32 i = 0; i < numCascade; i++) { tmpStr = bodyCascade[i]->folding(tmpStr, 0, nCount, from); nCount = tmpStr.getLength(); assert(from.getLength() == nCount); from.swap(to); for (sal_Int32& ix : asNonConstRange(to)) ix = from[ix]; } offset = to; return tmpStr; } } OUString SAL_CALL TransliterationImpl::transliterateString2String( const OUString& inStr, sal_Int32 startPos, sal_Int32 nCount ) { if (numCascade == 0) return inStr; else if (numCascade == 1) return bodyCascade[0]->transliterateString2String( inStr, startPos, nCount); else { OUString tmpStr = bodyCascade[0]->transliterateString2String(inStr, startPos, nCount); for (sal_Int32 i = 1; i < numCascade; i++) tmpStr = bodyCascade[i]->transliterateString2String(tmpStr, 0, tmpStr.getLength()); return tmpStr; } } OUString SAL_CALL TransliterationImpl::transliterateChar2String( sal_Unicode inChar ) { if (numCascade == 0) return OUString(&inChar, 1); else if (numCascade == 1) return bodyCascade[0]->transliterateChar2String( inChar); else { OUString tmpStr = bodyCascade[0]->transliterateChar2String(inChar); for (sal_Int32 i = 1; i < numCascade; i++) tmpStr = bodyCascade[i]->transliterateString2String(tmpStr, 0, tmpStr.getLength()); return tmpStr; } } sal_Unicode SAL_CALL TransliterationImpl::transliterateChar2Char( sal_Unicode inChar ) { sal_Unicode tmpChar = inChar; for (sal_Int32 i = 0; i < numCascade; i++) tmpChar = bodyCascade[i]->transliterateChar2Char(tmpChar); return tmpChar; } sal_Bool SAL_CALL TransliterationImpl::equals( const OUString& str1, sal_Int32 pos1, sal_Int32 nCount1, sal_Int32& nMatch1, const OUString& str2, sal_Int32 pos2, sal_Int32 nCount2, sal_Int32& nMatch2) { // since this is an API function make it user fail safe if ( nCount1 < 0 ) { pos1 += nCount1; nCount1 = -nCount1; } if ( nCount2 < 0 ) { pos2 += nCount2; nCount2 = -nCount2; } if ( !nCount1 || !nCount2 || pos1 >= str1.getLength() || pos2 >= str2.getLength() || pos1 < 0 || pos2 < 0 ) { nMatch1 = nMatch2 = 0; // two empty strings return true, else false return !nCount1 && !nCount2 && pos1 == str1.getLength() && pos2 == str2.getLength(); } if ( pos1 + nCount1 > str1.getLength() ) nCount1 = str1.getLength() - pos1; if ( pos2 + nCount2 > str2.getLength() ) nCount2 = str2.getLength() - pos2; if (caseignoreOnly && caseignore.is()) return caseignore->equals(str1, pos1, nCount1, nMatch1, str2, pos2, nCount2, nMatch2); Sequence offset1, offset2; OUString tmpStr1 = folding(str1, pos1, nCount1, offset1); OUString tmpStr2 = folding(str2, pos2, nCount2, offset2); // Length of offset1 and offset2 may still be 0 if there was no folding // necessary! const sal_Unicode *p1 = tmpStr1.getStr(); const sal_Unicode *p2 = tmpStr2.getStr(); sal_Int32 i, nLen = ::std::min( tmpStr1.getLength(), tmpStr2.getLength()); for (i = 0; i < nLen; ++i, ++p1, ++p2 ) { if (*p1 != *p2) { // return number of matched code points so far nMatch1 = (i < offset1.getLength()) ? offset1.getConstArray()[i] : i; nMatch2 = (i < offset2.getLength()) ? offset2.getConstArray()[i] : i; return false; } } // i==nLen if ( tmpStr1.getLength() != tmpStr2.getLength() ) { // return number of matched code points so far nMatch1 = (i <= offset1.getLength()) ? offset1.getConstArray()[i-1] + 1 : i; nMatch2 = (i <= offset2.getLength()) ? offset2.getConstArray()[i-1] + 1 : i; return false; } else { nMatch1 = nCount1; nMatch2 = nCount2; return true; } } Sequence< OUString > TransliterationImpl::getRange(const Sequence< OUString > &inStrs, const sal_Int32 length, sal_Int16 _numCascade) { if (_numCascade >= numCascade || ! bodyCascade[_numCascade].is()) return inStrs; sal_Int32 j_tmp = 0; constexpr sal_Int32 nMaxOutput = 2; const sal_Int32 nMaxOutputLength = nMaxOutput*length; std::vector ostr; ostr.reserve(nMaxOutputLength); for (sal_Int32 j = 0; j < length; j+=2) { const Sequence< OUString >& temp = bodyCascade[_numCascade]->transliterateRange(inStrs[j], inStrs[j+1]); for (const auto& rStr : temp) { if ( j_tmp++ >= nMaxOutputLength ) throw RuntimeException(); ostr.push_back(rStr); } } return getRange(comphelper::containerToSequence(ostr), j_tmp, ++_numCascade); } Sequence< OUString > SAL_CALL TransliterationImpl::transliterateRange( const OUString& str1, const OUString& str2 ) { if (numCascade == 1) return bodyCascade[0]->transliterateRange(str1, str2); Sequence< OUString > ostr{ str1, str2 }; return getRange(ostr, 2, 0); } sal_Int32 SAL_CALL TransliterationImpl::compareSubstring( const OUString& str1, sal_Int32 off1, sal_Int32 len1, const OUString& str2, sal_Int32 off2, sal_Int32 len2) { if (caseignoreOnly && caseignore.is()) return caseignore->compareSubstring(str1, off1, len1, str2, off2, len2); Sequence offset; OUString in_str1 = transliterate(str1, off1, len1, offset); OUString in_str2 = transliterate(str2, off2, len2, offset); const sal_Unicode* unistr1 = in_str1.getStr(); const sal_Unicode* unistr2 = in_str2.getStr(); sal_Int32 strlen1 = in_str1.getLength(); sal_Int32 strlen2 = in_str2.getLength(); while (strlen1 && strlen2) { if (*unistr1 != *unistr2) return *unistr1 > *unistr2 ? 1 : -1; unistr1++; unistr2++; strlen1--; strlen2--; } return strlen1 == strlen2 ? 0 : (strlen1 > strlen2 ? 1 : -1); } sal_Int32 SAL_CALL TransliterationImpl::compareString(const OUString& str1, const OUString& str2 ) { if (caseignoreOnly && caseignore.is()) return caseignore->compareString(str1, str2); else return compareSubstring(str1, 0, str1.getLength(), str2, 0, str2.getLength()); } void TransliterationImpl::clear() { for (sal_Int32 i = 0; i < numCascade; i++) if (bodyCascade[i].is()) bodyCascade[i].clear(); numCascade = 0; caseignore.clear(); caseignoreOnly = true; } namespace { /** structure to cache the last transliteration body used. */ struct TransBody { OUString Name; css::uno::Reference< css::i18n::XExtendedTransliteration > Body; }; } void TransliterationImpl::loadBody( OUString const &implName, Reference& body ) { assert(!implName.isEmpty()); static std::mutex transBodyMutex; std::unique_lock guard(transBodyMutex); static TransBody lastTransBody; if (implName != lastTransBody.Name) { lastTransBody.Body.set( mxContext->getServiceManager()->createInstanceWithContext(implName, mxContext), UNO_QUERY_THROW); lastTransBody.Name = implName; } body = lastTransBody.Body; } bool TransliterationImpl::loadModuleByName( std::u16string_view implName, Reference& body, const Locale& rLocale) { OUString cname = OUString::Concat(TRLT_IMPLNAME_PREFIX) + implName; loadBody(cname, body); if (body.is()) { body->loadModule(TransliterationModules(0), rLocale); // toUpper/toLoad need rLocale // if the module is ignore case/kana/width, load caseignore for equals/compareString mothed for (sal_Int16 i = 0; i < 3; i++) { if (o3tl::equalsAscii(implName, TMlist[i].implName)) { if (i == 0) // current module is caseignore body->loadModule(TMlist[0].tm, rLocale); // caseignore need to setup module name if (! caseignore.is()) { OUString bname = TRLT_IMPLNAME_PREFIX + OUString::createFromAscii(TMlist[0].implName); loadBody(bname, caseignore); } if (caseignore.is()) caseignore->loadModule(TMlist[i].tm, rLocale); return true; } } caseignoreOnly = false; // has other module than just ignore case/kana/width } return body.is(); } OUString SAL_CALL TransliterationImpl::getImplementationName() { return "com.sun.star.i18n.Transliteration"; } sal_Bool SAL_CALL TransliterationImpl::supportsService(const OUString& rServiceName) { return cppu::supportsService(this, rServiceName); } Sequence< OUString > SAL_CALL TransliterationImpl::getSupportedServiceNames() { return { "com.sun.star.i18n.Transliteration" }; } } extern "C" SAL_DLLPUBLIC_EXPORT css::uno::XInterface * com_sun_star_i18n_Transliteration_get_implementation( css::uno::XComponentContext *context, css::uno::Sequence const &) { return cppu::acquire(new i18npool::TransliterationImpl(context)); } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */