/* -*- 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 namespace { /** convert string to number with optional min and max values */ template bool convertNumber( T& rValue, const OUString& rString, T /*nMin*/ = -1, T /*nMax*/ = -1) { bool bNeg = false; rValue = 0; sal_Int32 nPos = 0; sal_Int32 nLen = rString.getLength(); // skip white space while( nPos < nLen && ' ' == rString[nPos] ) nPos++; if( nPos < nLen && '-' == rString[nPos] ) { bNeg = true; nPos++; } // get number while( nPos < nLen && '0' <= rString[nPos] && '9' >= rString[nPos] ) { // TODO: check overflow! rValue *= 10; rValue += (rString[nPos] - u'0'); nPos++; } if( bNeg ) rValue *= -1; return nPos == nLen; } // although the standard calls for fixed-length (zero-padded) tokens // (in their integer part), we are here liberal and allow shorter tokens // (when there are separators, else it is ambiguous). // Note that: // the token separator is OPTIONAL // empty string is a valid token! (to recognise hh or hhmm or hh:mm formats) // returns: success / failure // in case of failure, no reference argument is changed // arguments: // i_str: string to extract token from // index: index in i_str where to start tokenizing // after return, start of *next* token (if any) // if this was the last token, then the value is UNDEFINED // o_strInt: output; integer part of token // o_bFraction: output; was there a fractional part? // o_strFrac: output; fractional part of token bool impl_getISO8601TimeToken(const OUString &i_str, sal_Int32 &nPos, OUString &resInt, bool &bFraction, OUString &resFrac) { bFraction = false; // all tokens are of length 2 const sal_Int32 nEndPos = nPos + 2; const sal_Unicode c0 = '0'; const sal_Unicode c9 = '9'; const sal_Unicode sep = ':'; for (;nPos < nEndPos && nPos < i_str.getLength(); ++nPos) { const sal_Unicode c = i_str[nPos]; if (c == sep) return true; if (c < c0 || c > c9) return false; resInt += OUStringLiteral1(c); } if (nPos == i_str.getLength() || i_str[nPos] == sep) return true; if (i_str[nPos] == ',' || i_str[nPos] == '.') { bFraction = true; ++nPos; for (; nPos < i_str.getLength(); ++nPos) { const sal_Unicode c = i_str[nPos]; if (c == 'Z' || c == '+' || c == '-') { --nPos; // we don't want to skip the tz separator return true; } if (c == sep) // fractional part allowed only in *last* token return false; if (c < c0 || c > c9) return false; resFrac += OUStringLiteral1(c); } OSL_ENSURE(nPos == i_str.getLength(), "impl_getISO8601TimeToken internal error; expected to be at end of string"); return true; } if (i_str[nPos] == 'Z' || i_str[nPos] == '+' || i_str[nPos] == '-') { --nPos; // we don't want to skip the tz separator return true; } else return false; } inline bool getISO8601TimeToken(const OUString &i_str, sal_Int32 &io_index, OUString &o_strInt, bool &o_bFraction, OUString &o_strFrac) { OUString resInt; OUString resFrac; bool bFraction = false; sal_Int32 index = io_index; if(!impl_getISO8601TimeToken(i_str, index, resInt, bFraction, resFrac)) return false; else { io_index = index+1; o_strInt = resInt; o_strFrac = resFrac; o_bFraction = bFraction; return true; } } inline bool getISO8601TimeZoneToken(const OUString &i_str, sal_Int32 &io_index, OUString &o_strInt) { const sal_Unicode c0 = '0'; const sal_Unicode c9 = '9'; const sal_Unicode sep = ':'; if (i_str[io_index] == 'Z') // UTC timezone indicator { ++io_index; o_strInt = "Z"; return true; } else if (i_str[io_index] == '+' || i_str[io_index] == '-') // other timezones indicator { ++io_index; o_strInt.clear(); for (; io_index < i_str.getLength(); ++io_index) { const sal_Unicode c = i_str[io_index]; if ((c < c0 || c > c9) && c != sep) return false; o_strInt += OUStringLiteral1(c); } return true; } else return false; } } namespace utl { void typeConvert(const Date& _rDate, css::util::Date& _rOut) { _rOut.Day = _rDate.GetDay(); _rOut.Month = _rDate.GetMonth(); _rOut.Year = _rDate.GetYear(); } void typeConvert(const css::util::Date& _rDate, Date& _rOut) { _rOut = Date(_rDate.Day, _rDate.Month, _rDate.Year); } void typeConvert(const DateTime& _rDateTime, css::util::DateTime& _rOut) { _rOut.Year = _rDateTime.GetYear(); _rOut.Month = _rDateTime.GetMonth(); _rOut.Day = _rDateTime.GetDay(); _rOut.Hours = _rDateTime.GetHour(); _rOut.Minutes = _rDateTime.GetMin(); _rOut.Seconds = _rDateTime.GetSec(); _rOut.NanoSeconds = _rDateTime.GetNanoSec(); } void typeConvert(const css::util::DateTime& _rDateTime, DateTime& _rOut) { Date aDate(_rDateTime.Day, _rDateTime.Month, _rDateTime.Year); tools::Time aTime(_rDateTime.Hours, _rDateTime.Minutes, _rDateTime.Seconds, _rDateTime.NanoSeconds); _rOut = DateTime(aDate, aTime); } void extractDate(const css::util::DateTime& _rDateTime, css::util::Date& _rOut) { _rOut.Day = _rDateTime.Day; _rOut.Month = _rDateTime.Month; _rOut.Year = _rDateTime.Year; } OUString toISO8601(const css::util::DateTime& rDateTime) { OUStringBuffer rBuffer; rBuffer.append((sal_Int32) rDateTime.Year); rBuffer.append('-'); if( rDateTime.Month < 10 ) rBuffer.append('0'); rBuffer.append((sal_Int32) rDateTime.Month); rBuffer.append('-'); if( rDateTime.Day < 10 ) rBuffer.append('0'); rBuffer.append((sal_Int32) rDateTime.Day); if( rDateTime.NanoSeconds != 0 || rDateTime.Seconds != 0 || rDateTime.Minutes != 0 || rDateTime.Hours != 0 ) { rBuffer.append('T'); if( rDateTime.Hours < 10 ) rBuffer.append('0'); rBuffer.append((sal_Int32) rDateTime.Hours); rBuffer.append(':'); if( rDateTime.Minutes < 10 ) rBuffer.append('0'); rBuffer.append((sal_Int32) rDateTime.Minutes); rBuffer.append(':'); if( rDateTime.Seconds < 10 ) rBuffer.append('0'); rBuffer.append((sal_Int32) rDateTime.Seconds); if ( rDateTime.NanoSeconds > 0) { OSL_ENSURE(rDateTime.NanoSeconds < 1000000000,"NanoSeconds cannot be more than 999 999 999"); rBuffer.append(','); std::ostringstream ostr; ostr.fill('0'); ostr.width(9); ostr << rDateTime.NanoSeconds; rBuffer.append(OUString::createFromAscii(ostr.str().c_str())); } } return rBuffer.makeStringAndClear(); } /** convert ISO8601 DateTime String to util::DateTime */ bool ISO8601parseDateTime(const OUString &rString, css::util::DateTime& rDateTime) { bool bSuccess = true; rtl::OUString aDateStr, aTimeStr; css::util::Date aDate; css::util::Time aTime; sal_Int32 nPos = rString.indexOf( 'T' ); if ( nPos >= 0 ) { aDateStr = rString.copy( 0, nPos ); aTimeStr = rString.copy( nPos + 1 ); } else aDateStr = rString; // no separator: only date part bSuccess = ISO8601parseDate(aDateStr, aDate); if ( bSuccess && !aTimeStr.isEmpty() ) // time is optional { bSuccess = ISO8601parseTime(aTimeStr, aTime); } if (bSuccess) { rDateTime = css::util::DateTime(aTime.NanoSeconds, aTime.Seconds, aTime.Minutes, aTime.Hours, aDate.Day, aDate.Month, aDate.Year, false); } return bSuccess; } /** convert ISO8601 Date String to util::Date */ // TODO: supports only calendar dates YYYY-MM-DD // MISSING: calendar dates YYYYMMDD YYYY-MM // year, week date, ordinal date bool ISO8601parseDate(const OUString &aDateStr, css::util::Date& rDate) { bool bSuccess = true; sal_Int32 nYear = 1899; sal_Int32 nMonth = 12; sal_Int32 nDay = 30; const sal_Unicode* pStr = aDateStr.getStr(); sal_Int32 nDateTokens = 1; while ( *pStr ) { if ( *pStr == '-' ) nDateTokens++; pStr++; } if ( nDateTokens > 3 || aDateStr.isEmpty() ) bSuccess = false; else { sal_Int32 n = 0; if ( !convertNumber( nYear, aDateStr.getToken( 0, '-', n ), 0, 9999 ) ) bSuccess = false; if ( nDateTokens >= 2 ) if ( !convertNumber( nMonth, aDateStr.getToken( 0, '-', n ), 0, 12 ) ) bSuccess = false; if ( nDateTokens >= 3 ) if ( !convertNumber( nDay, aDateStr.getToken( 0, '-', n ), 0, 31 ) ) bSuccess = false; } if (bSuccess) { rDate.Year = (sal_uInt16)nYear; rDate.Month = (sal_uInt16)nMonth; rDate.Day = (sal_uInt16)nDay; } return bSuccess; } /** convert ISO8601 Time String to util::Time */ bool ISO8601parseTime(const OUString &aTimeStr, css::util::Time& rTime) { sal_Int32 nHour = 0; sal_Int32 nMin = 0; sal_Int32 nSec = 0; sal_Int32 nNanoSec = 0; sal_Int32 n = 0; OUString tokInt; OUString tokFrac; OUString tokTz; bool bFrac = false; // hours bool bSuccess = getISO8601TimeToken(aTimeStr, n, tokInt, bFrac, tokFrac); if (bSuccess) { if ( bFrac && n < aTimeStr.getLength()) // is it junk or the timezone? bSuccess = getISO8601TimeZoneToken(aTimeStr, n, tokTz); if (bSuccess && (bSuccess = convertNumber( nHour, tokInt, 0, 23 )) ) { if (bFrac) { sal_Int64 fracNumerator; if ( (bSuccess = convertNumber(fracNumerator, tokFrac)) ) { double frac = static_cast(fracNumerator) / pow(static_cast(10), static_cast(tokFrac.getLength())); // minutes OSL_ENSURE(frac < 1 && frac >= 0, "ISO8601parse internal error frac hours (of hours) not between 0 and 1"); frac *= 60; nMin = floor(frac); frac -= nMin; // seconds OSL_ENSURE(frac < 1 && frac >= 0, "ISO8601parse internal error frac minutes (of hours) not between 0 and 1"); frac *= 60; nSec = floor(frac); frac -= nSec; // nanoseconds OSL_ENSURE(frac < 1 && frac >= 0, "ISO8601parse internal error frac seconds (of hours) not between 0 and 1"); frac *= 1000000000; nNanoSec = ::rtl::math::round(frac); } goto end; } if(n >= aTimeStr.getLength()) goto end; } } // minutes if (bSuccess && (bSuccess = getISO8601TimeToken(aTimeStr, n, tokInt, bFrac, tokFrac))) { if ( bFrac && n < aTimeStr.getLength()) // is it junk or the timezone? bSuccess = getISO8601TimeZoneToken(aTimeStr, n, tokTz); if (bSuccess && (bSuccess = convertNumber( nMin, tokInt, 0, 59 )) ) { if (bFrac) { sal_Int64 fracNumerator; if ( (bSuccess = convertNumber(fracNumerator, tokFrac)) ) { double frac = static_cast(fracNumerator) / pow(static_cast(10), static_cast(tokFrac.getLength())); // seconds OSL_ENSURE(frac < 1 && frac >= 0, "ISO8601parse internal error frac minutes (of minutes) not between 0 and 1"); frac *= 60; nSec = floor(frac); frac -= nSec; // nanoseconds OSL_ENSURE(frac < 1 && frac >= 0, "ISO8601parse internal error frac seconds (of minutes) not between 0 and 1"); frac *= 1000000000; nNanoSec = ::rtl::math::round(frac); } goto end; } if(n >= aTimeStr.getLength()) goto end; } } // seconds if (bSuccess && (bSuccess = getISO8601TimeToken(aTimeStr, n, tokInt, bFrac, tokFrac))) { if (n < aTimeStr.getLength()) // is it junk or the timezone? bSuccess = getISO8601TimeZoneToken(aTimeStr, n, tokTz); // max 60 for leap seconds if (bSuccess && (bSuccess = convertNumber( nSec, tokInt, 0, 60 )) ) { if (bFrac) { sal_Int64 fracNumerator; if ( (bSuccess = convertNumber(fracNumerator, tokFrac)) ) { double frac = static_cast(fracNumerator) / pow(static_cast(10), static_cast(tokFrac.getLength())); // nanoseconds OSL_ENSURE(frac < 1 && frac >= 0, "ISO8601parse internal error frac seconds (of seconds) not between 0 and 1"); frac *= 1000000000; nNanoSec = ::rtl::math::round(frac); } goto end; } } } end: if (bSuccess) { // normalise time const int secondsOverflow = (nSec == 60) ? 61 : 60; if (nNanoSec == 1000000000) { nNanoSec = 0; ++nSec; } if(nSec == secondsOverflow) { nSec = 0; ++nMin; } if(nMin == 60) { nMin = 0; ++nHour; } if(!tokTz.isEmpty()) rTime.IsUTC = (tokTz == "Z"); rTime.Hours = (sal_uInt16)nHour; rTime.Minutes = (sal_uInt16)nMin; rTime.Seconds = (sal_uInt16)nSec; rTime.NanoSeconds = nNanoSec; } return bSuccess; } } // namespace utl /* vim:set shiftwidth=4 softtabstop=4 expandtab: */