summaryrefslogtreecommitdiff
path: root/i18npool/source/collator/collatorImpl.cxx
diff options
context:
space:
mode:
Diffstat (limited to 'i18npool/source/collator/collatorImpl.cxx')
-rw-r--r--i18npool/source/collator/collatorImpl.cxx260
1 files changed, 260 insertions, 0 deletions
diff --git a/i18npool/source/collator/collatorImpl.cxx b/i18npool/source/collator/collatorImpl.cxx
new file mode 100644
index 000000000000..bdd28a255d07
--- /dev/null
+++ b/i18npool/source/collator/collatorImpl.cxx
@@ -0,0 +1,260 @@
+/*************************************************************************
+ *
+ * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
+ *
+ * Copyright 2000, 2010 Oracle and/or its affiliates.
+ *
+ * OpenOffice.org - a multi-platform office productivity suite
+ *
+ * This file is part of OpenOffice.org.
+ *
+ * OpenOffice.org is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Lesser General Public License version 3
+ * only, as published by the Free Software Foundation.
+ *
+ * OpenOffice.org is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU Lesser General Public License version 3 for more details
+ * (a copy is included in the LICENSE file that accompanied this code).
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * version 3 along with OpenOffice.org. If not, see
+ * <http://www.openoffice.org/license.html>
+ * for a copy of the LGPLv3 License.
+ *
+ ************************************************************************/
+
+// MARKER(update_precomp.py): autogen include statement, do not remove
+#include "precompiled_i18npool.hxx"
+
+#include <collatorImpl.hxx>
+#include <com/sun/star/i18n/CollatorOptions.hpp>
+#include <rtl/ustrbuf.hxx>
+
+using namespace com::sun::star;
+using namespace com::sun::star::lang;
+using namespace com::sun::star::uno;
+using namespace rtl;
+
+namespace com { namespace sun { namespace star { namespace i18n {
+
+CollatorImpl::CollatorImpl( const Reference < XMultiServiceFactory >& rxMSF ) : xMSF(rxMSF)
+{
+ if ( rxMSF.is()) {
+ Reference < XInterface > xI =
+ xMSF->createInstance( OUString::createFromAscii("com.sun.star.i18n.LocaleData"));
+ if ( xI.is() )
+ xI->queryInterface(::getCppuType((const Reference< XLocaleData>*)0)) >>= localedata;
+ }
+ cachedItem = NULL;
+}
+
+CollatorImpl::~CollatorImpl()
+{
+ // Clear lookuptable
+ for (size_t l = 0; l < lookupTable.size(); l++)
+ delete lookupTable[l];
+ lookupTable.clear();
+}
+
+sal_Int32 SAL_CALL
+CollatorImpl::compareSubstring( const OUString& str1, sal_Int32 off1, sal_Int32 len1,
+ const OUString& str2, sal_Int32 off2, sal_Int32 len2) throw(RuntimeException)
+{
+ if (cachedItem)
+ return cachedItem->xC->compareSubstring(str1, off1, len1, str2, off2, len2);
+
+ sal_Unicode *unistr1 = (sal_Unicode*) str1.getStr() + off1;
+ sal_Unicode *unistr2 = (sal_Unicode*) str2.getStr() + off2;
+ for (int i = 0; i < len1 && i < len2; i++)
+ if (unistr1[i] != unistr2[i])
+ return unistr1[i] < unistr2[i] ? -1 : 1;
+ return len1 == len2 ? 0 : (len1 < len2 ? -1 : 1);
+}
+
+sal_Int32 SAL_CALL
+CollatorImpl::compareString( const OUString& in_str1, const OUString& in_str2) throw(RuntimeException)
+{
+ if (cachedItem)
+ return cachedItem->xC->compareString(in_str1, in_str2);
+
+ return CollatorImpl::compareSubstring(in_str1, 0, in_str1.getLength(), in_str2, 0, in_str2.getLength());
+}
+
+
+sal_Int32 SAL_CALL
+CollatorImpl::loadDefaultCollator(const lang::Locale& rLocale, sal_Int32 collatorOptions) throw(RuntimeException)
+{
+ const Sequence< Implementation > &imp = localedata->getCollatorImplementations(rLocale);
+ for (sal_Int16 i = 0; i < imp.getLength(); i++)
+ if (imp[i].isDefault)
+ return loadCollatorAlgorithm(imp[i].unoID, rLocale, collatorOptions);
+
+ throw RuntimeException(); // not default is defined
+ //return 0;
+}
+
+sal_Int32 SAL_CALL
+CollatorImpl::loadCollatorAlgorithm(const OUString& impl, const lang::Locale& rLocale, sal_Int32 collatorOptions)
+ throw(RuntimeException)
+{
+ if (! cachedItem || ! cachedItem->equals(rLocale, impl))
+ loadCachedCollator(rLocale, impl);
+
+ if (cachedItem)
+ cachedItem->xC->loadCollatorAlgorithm(cachedItem->algorithm, nLocale = rLocale, collatorOptions);
+ else
+ throw RuntimeException(); // impl could not be loaded
+
+ return 0;
+}
+
+void SAL_CALL
+CollatorImpl::loadCollatorAlgorithmWithEndUserOption(const OUString& impl, const lang::Locale& rLocale,
+ const Sequence< sal_Int32 >& collatorOptions) throw(RuntimeException)
+{
+ sal_Int32 options = 0;
+ for (sal_Int32 i = 0; i < collatorOptions.getLength(); i++)
+ options |= collatorOptions[i];
+ loadCollatorAlgorithm(impl, rLocale, options);
+}
+
+Sequence< OUString > SAL_CALL
+CollatorImpl::listCollatorAlgorithms( const lang::Locale& rLocale ) throw(RuntimeException)
+{
+ nLocale = rLocale;
+ const Sequence< Implementation > &imp = localedata->getCollatorImplementations(rLocale);
+ Sequence< OUString > list(imp.getLength());
+
+ for (sal_Int32 i = 0; i < imp.getLength(); i++) {
+ //if the current algorithm is default and the position is not on the first one, then switch
+ if (imp[i].isDefault && i) {
+ list[i] = list[0];
+ list[0] = imp[i].unoID;
+ }
+ else
+ list[i] = imp[i].unoID;
+ }
+ return list;
+}
+
+Sequence< sal_Int32 > SAL_CALL
+CollatorImpl::listCollatorOptions( const OUString& /*collatorAlgorithmName*/ ) throw(RuntimeException)
+{
+ Sequence< OUString > option_str = localedata->getCollationOptions(nLocale);
+ Sequence< sal_Int32 > option_int(option_str.getLength());
+
+ for (sal_Int32 i = 0; i < option_str.getLength(); i++)
+ option_int[i] =
+ option_str[i].equalsAscii("IGNORE_CASE") ? CollatorOptions::CollatorOptions_IGNORE_CASE :
+ option_str[i].equalsAscii("IGNORE_KANA") ? CollatorOptions::CollatorOptions_IGNORE_KANA :
+ option_str[i].equalsAscii("IGNORE_WIDTH") ? CollatorOptions::CollatorOptions_IGNORE_WIDTH : 0;
+
+ return option_int;
+}
+
+sal_Bool SAL_CALL
+CollatorImpl::createCollator(const lang::Locale& rLocale, const OUString& serviceName, const OUString& rSortAlgorithm)
+ throw(RuntimeException)
+{
+ for (size_t l = 0; l < lookupTable.size(); l++) {
+ cachedItem = lookupTable[l];
+ if (cachedItem->service.equals(serviceName)) {// cross locale sharing
+ lookupTable.push_back(cachedItem = new lookupTableItem(rLocale, rSortAlgorithm, serviceName, cachedItem->xC));
+ return sal_True;
+ }
+ }
+ if (xMSF.is()) {
+ Reference < XInterface > xI =
+ xMSF->createInstance(OUString::createFromAscii("com.sun.star.i18n.Collator_") + serviceName);
+
+ if (xI.is()) {
+ Reference < XCollator > xC;
+ xI->queryInterface( getCppuType((const Reference< XCollator>*)0) ) >>= xC;
+ if (xC.is()) {
+ lookupTable.push_back(cachedItem = new lookupTableItem(rLocale, rSortAlgorithm, serviceName, xC));
+ return sal_True;
+ }
+ }
+ return sal_False;
+ }
+ throw RuntimeException();
+}
+
+void SAL_CALL
+CollatorImpl::loadCachedCollator(const lang::Locale& rLocale, const OUString& rSortAlgorithm)
+ throw(RuntimeException)
+{
+ for (size_t i = 0; i < lookupTable.size(); i++) {
+ cachedItem = lookupTable[i];
+ if (cachedItem->equals(rLocale, rSortAlgorithm)) {
+ return;
+ }
+ }
+
+ static sal_Unicode under = (sal_Unicode) '_';
+ static OUString tw(OUString::createFromAscii("TW"));
+ static OUString unicode(OUString::createFromAscii("Unicode"));
+
+ sal_Int32 l = rLocale.Language.getLength();
+ sal_Int32 c = rLocale.Country.getLength();
+ sal_Int32 v = rLocale.Variant.getLength();
+ sal_Int32 a = rSortAlgorithm.getLength();
+ OUStringBuffer aBuf(l+c+v+a+4);
+
+ if ((l > 0 && c > 0 && v > 0 && a > 0 &&
+ // load service with name <base>_<lang>_<country>_<varian>_<algorithm>
+ createCollator(rLocale, aBuf.append(rLocale.Language).append(under).append(rLocale.Country).append(
+ under).append(rLocale.Variant).append(under).append(rSortAlgorithm).makeStringAndClear(),
+ rSortAlgorithm)) ||
+ (l > 0 && c > 0 && a > 0 &&
+ // load service with name <base>_<lang>_<country>_<algorithm>
+ createCollator(rLocale, aBuf.append(rLocale.Language).append(under).append(rLocale.Country).append(
+ under).append(rSortAlgorithm).makeStringAndClear(), rSortAlgorithm)) ||
+ (l > 0 && c > 0 && a > 0 && rLocale.Language.equalsAscii("zh") &&
+ (rLocale.Country.equalsAscii("HK") ||
+ rLocale.Country.equalsAscii("MO")) &&
+ // if the country code is HK or MO, one more step to try TW.
+ createCollator(rLocale, aBuf.append(rLocale.Language).append(under).append(tw).append(under).append(
+ rSortAlgorithm).makeStringAndClear(), rSortAlgorithm)) ||
+ (l > 0 && a > 0 &&
+ // load service with name <base>_<lang>_<algorithm>
+ createCollator(rLocale, aBuf.append(rLocale.Language).append(under).append(rSortAlgorithm).makeStringAndClear(),
+ rSortAlgorithm)) ||
+ // load service with name <base>_<algorithm>
+ (a > 0 &&
+ createCollator(rLocale, rSortAlgorithm, rSortAlgorithm)) ||
+ // load default service with name <base>_Unicode
+ createCollator(rLocale, unicode, rSortAlgorithm)) {
+ return;
+ } else {
+ cachedItem = NULL;
+ throw RuntimeException(); // could not load any service
+ }
+}
+
+const sal_Char cCollator[] = "com.sun.star.i18n.Collator";
+
+OUString SAL_CALL
+CollatorImpl::getImplementationName() throw( RuntimeException )
+{
+ return OUString::createFromAscii(cCollator);
+}
+
+sal_Bool SAL_CALL
+CollatorImpl::supportsService(const OUString& rServiceName)
+ throw( RuntimeException )
+{
+ return rServiceName.equalsAscii(cCollator);
+}
+
+Sequence< OUString > SAL_CALL
+CollatorImpl::getSupportedServiceNames() throw( RuntimeException )
+{
+ Sequence< OUString > aRet(1);
+ aRet[0] = OUString::createFromAscii(cCollator);
+ return aRet;
+}
+
+} } } }