LCOV - code coverage report
Current view: top level - forms/source/misc - limitedformats.cxx (source / functions) Hit Total Coverage
Test: commit c8344322a7af75b84dd3ca8f78b05543a976dfd5 Lines: 109 126 86.5 %
Date: 2015-06-13 12:38:46 Functions: 13 14 92.9 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
       2             : /*
       3             :  * This file is part of the LibreOffice project.
       4             :  *
       5             :  * This Source Code Form is subject to the terms of the Mozilla Public
       6             :  * License, v. 2.0. If a copy of the MPL was not distributed with this
       7             :  * file, You can obtain one at http://mozilla.org/MPL/2.0/.
       8             :  *
       9             :  * This file incorporates work covered by the following license notice:
      10             :  *
      11             :  *   Licensed to the Apache Software Foundation (ASF) under one or more
      12             :  *   contributor license agreements. See the NOTICE file distributed
      13             :  *   with this work for additional information regarding copyright
      14             :  *   ownership. The ASF licenses this file to you under the Apache
      15             :  *   License, Version 2.0 (the "License"); you may not use this file
      16             :  *   except in compliance with the License. You may obtain a copy of
      17             :  *   the License at http://www.apache.org/licenses/LICENSE-2.0 .
      18             :  */
      19             : 
      20             : #include "limitedformats.hxx"
      21             : #include "services.hxx"
      22             : #include <osl/diagnose.h>
      23             : #include <comphelper/types.hxx>
      24             : #include <comphelper/extract.hxx>
      25             : #include <com/sun/star/form/FormComponentType.hpp>
      26             : #include <com/sun/star/util/NumberFormatsSupplier.hpp>
      27             : 
      28             : 
      29             : namespace frm
      30             : {
      31             : 
      32             : 
      33             :     using namespace ::com::sun::star::uno;
      34             :     using namespace ::com::sun::star::util;
      35             :     using namespace ::com::sun::star::lang;
      36             :     using namespace ::com::sun::star::form;
      37             :     using namespace ::com::sun::star::beans;
      38             : 
      39             :     sal_Int32                               OLimitedFormats::s_nInstanceCount(0);
      40          46 :     ::osl::Mutex                            OLimitedFormats::s_aMutex;
      41          46 :     Reference< XNumberFormatsSupplier >     OLimitedFormats::s_xStandardFormats;
      42             : 
      43             : 
      44             :     //=
      45             : 
      46             : 
      47             :     enum LocaleType
      48             :     {
      49             :         ltEnglishUS,
      50             :         ltGerman,
      51             :         ltSystem
      52             :     };
      53             : 
      54             : 
      55         438 :     static const Locale& getLocale(LocaleType _eType)
      56             :     {
      57         438 :         static const Locale s_aEnglishUS( OUString("en"), OUString("us"), OUString() );
      58         438 :         static const Locale s_aGerman( OUString("de"), OUString("DE"), OUString() );
      59         438 :         static const OUString s_sEmptyString;
      60         438 :         static const Locale s_aSystem( s_sEmptyString, s_sEmptyString, s_sEmptyString );
      61             : 
      62         438 :         switch (_eType)
      63             :         {
      64             :             case ltEnglishUS:
      65         249 :                 return s_aEnglishUS;
      66             : 
      67             :             case ltGerman:
      68         189 :                 return s_aGerman;
      69             : 
      70             :             case ltSystem:
      71           0 :                 return s_aSystem;
      72             :         }
      73             : 
      74             :         OSL_FAIL("getLocale: invalid enum value!");
      75           0 :         return s_aSystem;
      76             :     }
      77             : 
      78             : 
      79             :     struct FormatEntry
      80             :     {
      81             :         const sal_Char* pDescription;
      82             :         sal_Int32       nKey;
      83             :         LocaleType      eLocale;
      84             :     };
      85             : 
      86             : 
      87         158 :     static FormatEntry* lcl_getFormatTable(sal_Int16 nTableId)
      88             :     {
      89         158 :         switch (nTableId)
      90             :         {
      91             :             case FormComponentType::TIMEFIELD:
      92             :             {
      93             :                 static FormatEntry s_aFormats[] = {
      94             :                     { "HH:MM", -1, ltEnglishUS },
      95             :                     { "HH:MM:SS", -1, ltEnglishUS },
      96             :                     { "HH:MM AM/PM", -1, ltEnglishUS },
      97             :                     { "HH:MM:SS AM/PM", -1, ltEnglishUS },
      98             :                     { NULL, -1, ltSystem }
      99             :                 };
     100          58 :                 return s_aFormats;
     101             :             }
     102             :             case FormComponentType::DATEFIELD:
     103             :             {
     104             :                 static FormatEntry s_aFormats[] = {
     105             :                     { "T-M-JJ", -1, ltGerman },
     106             :                     { "TT-MM-JJ", -1, ltGerman },
     107             :                     { "TT-MM-JJJJ", -1, ltGerman },
     108             :                     { "NNNNT. MMMM JJJJ", -1, ltGerman },
     109             : 
     110             :                     { "DD/MM/YY", -1, ltEnglishUS },
     111             :                     { "MM/DD/YY", -1, ltEnglishUS },
     112             :                     { "YY/MM/DD", -1, ltEnglishUS },
     113             :                     { "DD/MM/YYYY", -1, ltEnglishUS },
     114             :                     { "MM/DD/YYYY", -1, ltEnglishUS },
     115             :                     { "YYYY/MM/DD", -1, ltEnglishUS },
     116             : 
     117             :                     { "JJ-MM-TT", -1, ltGerman },
     118             :                     { "JJJJ-MM-TT", -1, ltGerman },
     119             : 
     120             :                     { NULL, -1, ltSystem }
     121             :                 };
     122         100 :                 return s_aFormats;
     123             :             }
     124             :         }
     125             : 
     126             :         OSL_FAIL("lcl_getFormatTable: invalid id!");
     127           0 :         return NULL;
     128             :     }
     129             : 
     130          48 :     OLimitedFormats::OLimitedFormats(const Reference< XComponentContext >& _rxContext, const sal_Int16 _nClassId)
     131             :         :m_nFormatEnumPropertyHandle(-1)
     132          48 :         ,m_nTableId(_nClassId)
     133             :     {
     134             :         OSL_ENSURE(_rxContext.is(), "OLimitedFormats::OLimitedFormats: invalid service factory!");
     135          48 :         acquireSupplier(_rxContext);
     136          48 :         ensureTableInitialized(m_nTableId);
     137          48 :     }
     138             : 
     139             : 
     140          92 :     OLimitedFormats::~OLimitedFormats()
     141             :     {
     142          46 :         releaseSupplier();
     143          46 :     }
     144             : 
     145             : 
     146          48 :     void OLimitedFormats::ensureTableInitialized(const sal_Int16 _nTableId)
     147             :     {
     148          48 :         FormatEntry* pFormatTable = lcl_getFormatTable(_nTableId);
     149          48 :         if (-1 == pFormatTable->nKey)
     150             :         {
     151          25 :             ::osl::MutexGuard aGuard(s_aMutex);
     152          25 :             if (-1 == pFormatTable->nKey)
     153             :             {
     154             :                 // initialize the keys
     155          25 :                 Reference<XNumberFormats> xStandardFormats;
     156          25 :                 if (s_xStandardFormats.is())
     157          25 :                     xStandardFormats = s_xStandardFormats->getNumberFormats();
     158             :                 OSL_ENSURE(xStandardFormats.is(), "OLimitedFormats::ensureTableInitialized: don't have a formats supplier!");
     159             : 
     160          25 :                 if (xStandardFormats.is())
     161             :                 {
     162             :                     // loop through the table
     163          25 :                     FormatEntry* pLoopFormats = pFormatTable;
     164         318 :                     while (pLoopFormats->pDescription)
     165             :                     {
     166             :                         // get the key for the description
     167         268 :                         pLoopFormats->nKey = xStandardFormats->queryKey(
     168             :                             OUString::createFromAscii(pLoopFormats->pDescription),
     169         268 :                             getLocale(pLoopFormats->eLocale),
     170             :                             sal_False
     171         536 :                         );
     172             : 
     173         268 :                         if (-1 == pLoopFormats->nKey)
     174             :                         {
     175         147 :                             pLoopFormats->nKey = xStandardFormats->addNew(
     176             :                                 OUString::createFromAscii(pLoopFormats->pDescription),
     177         147 :                                 getLocale(pLoopFormats->eLocale)
     178         294 :                             );
     179             : #ifdef DBG_UTIL
     180             :                             try
     181             :                             {
     182             :                                 xStandardFormats->getByKey(pLoopFormats->nKey);
     183             :                             }
     184             :                             catch(const Exception&)
     185             :                             {
     186             :                                 OSL_FAIL("OLimitedFormats::ensureTableInitialized: adding the key to the formats collection failed!");
     187             :                             }
     188             : #endif
     189             :                         }
     190             : 
     191             :                         // next
     192         268 :                         ++pLoopFormats;
     193             :                     }
     194          25 :                 }
     195          25 :             }
     196             :         }
     197          48 :     }
     198             : 
     199             : 
     200          44 :     void OLimitedFormats::clearTable(const sal_Int16 _nTableId)
     201             :     {
     202          44 :         ::osl::MutexGuard aGuard(s_aMutex);
     203          44 :         FormatEntry* pFormats = lcl_getFormatTable(_nTableId);
     204          44 :         FormatEntry* pResetLoop = pFormats;
     205         440 :         while (pResetLoop->pDescription)
     206             :         {
     207         352 :             pResetLoop->nKey = -1;
     208         352 :             ++pResetLoop;
     209          44 :         }
     210          44 :     }
     211             : 
     212             : 
     213          94 :     void OLimitedFormats::setAggregateSet(const Reference< XFastPropertySet >& _rxAggregate, sal_Int32 _nOriginalPropertyHandle)
     214             :     {
     215             :         // changes (NULL -> not NULL) and (not NULL -> NULL) are allowed
     216             :         OSL_ENSURE(!m_xAggregate.is() || !_rxAggregate.is(), "OLimitedFormats::setAggregateSet: already have an aggregate!");
     217             :         OSL_ENSURE(_rxAggregate.is() || m_xAggregate.is(), "OLimitedFormats::setAggregateSet: invalid new aggregate!");
     218             : 
     219          94 :         m_xAggregate = _rxAggregate;
     220          94 :         m_nFormatEnumPropertyHandle = _nOriginalPropertyHandle;
     221             : #ifdef DBG_UTIL
     222             :         if (m_xAggregate.is())
     223             :         {
     224             :             try
     225             :             {
     226             :                 m_xAggregate->getFastPropertyValue(m_nFormatEnumPropertyHandle);
     227             :             }
     228             :             catch(const Exception&)
     229             :             {
     230             :                 OSL_FAIL("OLimitedFormats::setAggregateSet: invalid handle!");
     231             :             }
     232             :         }
     233             : #endif
     234          94 :     }
     235             : 
     236             : 
     237          64 :     void OLimitedFormats::getFormatKeyPropertyValue( Any& _rValue ) const
     238             :     {
     239          64 :         _rValue.clear();
     240             : 
     241             :         OSL_ENSURE(m_xAggregate.is() && (-1 != m_nFormatEnumPropertyHandle), "OLimitedFormats::getFormatKeyPropertyValue: not initialized!");
     242          64 :         if (m_xAggregate.is())
     243             :         {
     244             :             // get the aggregate's enum property value
     245          64 :             Any aEnumPropertyValue = m_xAggregate->getFastPropertyValue(m_nFormatEnumPropertyHandle);
     246          64 :             sal_Int32 nValue = -1;
     247          64 :             ::cppu::enum2int(nValue, aEnumPropertyValue);
     248             : 
     249             :             // get the translation table
     250          64 :             const FormatEntry* pFormats = lcl_getFormatTable(m_nTableId);
     251             : 
     252             :             // seek to the nValue'th entry
     253          64 :             sal_Int32 nLookup = 0;
     254         236 :             for (   ;
     255         344 :                     (NULL != pFormats->pDescription) && (nLookup < nValue);
     256             :                     ++pFormats, ++nLookup
     257             :                 )
     258             :                 ;
     259             :             OSL_ENSURE(NULL != pFormats->pDescription, "OLimitedFormats::getFormatKeyPropertyValue: did not find the value!");
     260          64 :             if (pFormats->pDescription)
     261          64 :                 _rValue <<= pFormats->nKey;
     262             :         }
     263             : 
     264             :         // TODO: should use a standard format for the control type we're working for
     265          64 :     }
     266             : 
     267             : 
     268           2 :     bool OLimitedFormats::convertFormatKeyPropertyValue(Any& _rConvertedValue, Any& _rOldValue, const Any& _rNewValue)
     269             :     {
     270             :         OSL_ENSURE(m_xAggregate.is() && (-1 != m_nFormatEnumPropertyHandle), "OLimitedFormats::convertFormatKeyPropertyValue: not initialized!");
     271             : 
     272           2 :         if (m_xAggregate.is())
     273             :         {
     274             :             // the new format key to set
     275           2 :             sal_Int32 nNewFormat = 0;
     276           2 :             if (!(_rNewValue >>= nNewFormat))
     277           0 :                 throw IllegalArgumentException();
     278             : 
     279             :             // get the old (enum) value from the aggregate
     280           2 :             Any aEnumPropertyValue = m_xAggregate->getFastPropertyValue(m_nFormatEnumPropertyHandle);
     281           2 :             sal_Int32 nOldEnumValue = -1;
     282           2 :             ::cppu::enum2int(nOldEnumValue, aEnumPropertyValue);
     283             : 
     284             :             // get the translation table
     285           2 :             const FormatEntry* pFormats = lcl_getFormatTable(m_nTableId);
     286             : 
     287           2 :             _rOldValue.clear();
     288           2 :             _rConvertedValue.clear();
     289             : 
     290             :             // look for the entry with the given format key
     291           2 :             sal_Int32 nTablePosition = 0;
     292          36 :             for (   ;
     293          34 :                     (NULL != pFormats->pDescription) && (nNewFormat != pFormats->nKey);
     294             :                     ++pFormats, ++nTablePosition
     295             :                 )
     296             :             {
     297          16 :                 if (nTablePosition == nOldEnumValue)
     298           2 :                     _rOldValue <<= pFormats->nKey;
     299             :             }
     300             : 
     301           2 :             bool bFoundIt = (NULL != pFormats->pDescription);
     302           2 :             bool bModified = false;
     303           2 :             if (bFoundIt)
     304             :             {
     305           0 :                 _rConvertedValue <<= (sal_Int16)nTablePosition;
     306           0 :                 bModified = nTablePosition != nOldEnumValue;
     307             :             }
     308             : 
     309           2 :             if (!_rOldValue.hasValue())
     310             :             {   // did not reach the end of the table (means we found nNewFormat)
     311             :                 // -> go to the end to ensure that _rOldValue is set
     312           0 :                 while (pFormats->pDescription)
     313             :                 {
     314           0 :                     if (nTablePosition == nOldEnumValue)
     315             :                     {
     316           0 :                         _rOldValue <<= pFormats->nKey;
     317           0 :                         break;
     318             :                     }
     319             : 
     320           0 :                     ++pFormats;
     321           0 :                     ++nTablePosition;
     322             :                 }
     323             :             }
     324             : 
     325             :             OSL_ENSURE(_rOldValue.hasValue(), "OLimitedFormats::convertFormatKeyPropertyValue: did not find the old enum value in the table!");
     326             : 
     327           2 :             if (!bFoundIt)
     328             :             {   // somebody gave us a format which we can't translate
     329           2 :                 OUString sMessage ("This control supports only a very limited number of formats.");
     330           2 :                 throw IllegalArgumentException(sMessage, NULL, 2);
     331             :             }
     332             : 
     333           2 :             return bModified;
     334             :         }
     335             : 
     336           0 :         return false;
     337             :     }
     338             : 
     339             : 
     340           0 :     void OLimitedFormats::setFormatKeyPropertyValue( const Any& _rNewValue )
     341             :     {
     342             :         OSL_ENSURE(m_xAggregate.is() && (-1 != m_nFormatEnumPropertyHandle), "OLimitedFormats::setFormatKeyPropertyValue: not initialized!");
     343             : 
     344           0 :         if (m_xAggregate.is())
     345             :         {   // this is to be called after convertFormatKeyPropertyValue, where
     346             :             // we translated the format key into an enum value.
     347             :             // So now we can simply forward this enum value to our aggregate
     348           0 :             m_xAggregate->setFastPropertyValue(m_nFormatEnumPropertyHandle, _rNewValue);
     349             :         }
     350           0 :     }
     351             : 
     352             : 
     353          48 :     void OLimitedFormats::acquireSupplier(const Reference< XComponentContext >& _rxContext)
     354             :     {
     355          48 :         ::osl::MutexGuard aGuard(s_aMutex);
     356          48 :         if (1 == ++s_nInstanceCount)
     357             :         {   // create the standard formatter
     358          23 :             s_xStandardFormats = NumberFormatsSupplier::createWithLocale(_rxContext, getLocale(ltEnglishUS));
     359          48 :         }
     360          48 :     }
     361             : 
     362             : 
     363          46 :     void OLimitedFormats::releaseSupplier()
     364             :     {
     365          46 :         ::osl::MutexGuard aGuard(s_aMutex);
     366          46 :         if (0 == --s_nInstanceCount)
     367             :         {
     368          22 :             ::comphelper::disposeComponent(s_xStandardFormats);
     369          22 :             s_xStandardFormats = NULL;
     370             : 
     371          22 :             clearTable(FormComponentType::TIMEFIELD);
     372          22 :             clearTable(FormComponentType::DATEFIELD);
     373          46 :         }
     374          46 :     }
     375             : 
     376             : 
     377         138 : }   // namespace frm
     378             : 
     379             : 
     380             : /* vim:set shiftwidth=4 softtabstop=4 expandtab: */

Generated by: LCOV version 1.11