Home | History | Annotate | Download | only in forms
      1 /*
      2  * Copyright (C) 2010 Google Inc. All rights reserved.
      3  *
      4  * Redistribution and use in source and binary forms, with or without
      5  * modification, are permitted provided that the following conditions are
      6  * met:
      7  *
      8  *     * Redistributions of source code must retain the above copyright
      9  * notice, this list of conditions and the following disclaimer.
     10  *     * Redistributions in binary form must reproduce the above
     11  * copyright notice, this list of conditions and the following disclaimer
     12  * in the documentation and/or other materials provided with the
     13  * distribution.
     14  *     * Neither the name of Google Inc. nor the names of its
     15  * contributors may be used to endorse or promote products derived from
     16  * this software without specific prior written permission.
     17  *
     18  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
     19  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
     20  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
     21  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
     22  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
     23  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
     24  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
     25  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
     26  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
     27  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
     28  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
     29  */
     30 
     31 #include "config.h"
     32 #include "core/html/forms/BaseDateAndTimeInputType.h"
     33 
     34 #include <limits>
     35 #include "core/html/HTMLInputElement.h"
     36 #include "platform/text/PlatformLocale.h"
     37 #include "wtf/CurrentTime.h"
     38 #include "wtf/DateMath.h"
     39 #include "wtf/MathExtras.h"
     40 #include "wtf/text/WTFString.h"
     41 
     42 namespace blink {
     43 
     44 using blink::WebLocalizedString;
     45 using namespace HTMLNames;
     46 
     47 static const int msecPerMinute = 60 * 1000;
     48 static const int msecPerSecond = 1000;
     49 
     50 double BaseDateAndTimeInputType::valueAsDate() const
     51 {
     52     return valueAsDouble();
     53 }
     54 
     55 void BaseDateAndTimeInputType::setValueAsDate(double value, ExceptionState&) const
     56 {
     57     element().setValue(serializeWithMilliseconds(value));
     58 }
     59 
     60 double BaseDateAndTimeInputType::valueAsDouble() const
     61 {
     62     const Decimal value = parseToNumber(element().value(), Decimal::nan());
     63     return value.isFinite() ? value.toDouble() : DateComponents::invalidMilliseconds();
     64 }
     65 
     66 void BaseDateAndTimeInputType::setValueAsDouble(double newValue, TextFieldEventBehavior eventBehavior, ExceptionState& exceptionState) const
     67 {
     68     setValueAsDecimal(Decimal::fromDouble(newValue), eventBehavior, exceptionState);
     69 }
     70 
     71 bool BaseDateAndTimeInputType::typeMismatchFor(const String& value) const
     72 {
     73     return !value.isEmpty() && !parseToDateComponents(value, 0);
     74 }
     75 
     76 bool BaseDateAndTimeInputType::typeMismatch() const
     77 {
     78     return typeMismatchFor(element().value());
     79 }
     80 
     81 String BaseDateAndTimeInputType::rangeOverflowText(const Decimal& maximum) const
     82 {
     83     return locale().queryString(WebLocalizedString::ValidationRangeOverflowDateTime, localizeValue(serialize(maximum)));
     84 }
     85 
     86 String BaseDateAndTimeInputType::rangeUnderflowText(const Decimal& minimum) const
     87 {
     88     return locale().queryString(WebLocalizedString::ValidationRangeUnderflowDateTime, localizeValue(serialize(minimum)));
     89 }
     90 
     91 Decimal BaseDateAndTimeInputType::defaultValueForStepUp() const
     92 {
     93     return Decimal::fromDouble(convertToLocalTime(currentTimeMS()));
     94 }
     95 
     96 bool BaseDateAndTimeInputType::isSteppable() const
     97 {
     98     return true;
     99 }
    100 
    101 Decimal BaseDateAndTimeInputType::parseToNumber(const String& source, const Decimal& defaultValue) const
    102 {
    103     DateComponents date;
    104     if (!parseToDateComponents(source, &date))
    105         return defaultValue;
    106     double msec = date.millisecondsSinceEpoch();
    107     ASSERT(std::isfinite(msec));
    108     return Decimal::fromDouble(msec);
    109 }
    110 
    111 bool BaseDateAndTimeInputType::parseToDateComponents(const String& source, DateComponents* out) const
    112 {
    113     if (source.isEmpty())
    114         return false;
    115     DateComponents ignoredResult;
    116     if (!out)
    117         out = &ignoredResult;
    118     return parseToDateComponentsInternal(source, out);
    119 }
    120 
    121 String BaseDateAndTimeInputType::serialize(const Decimal& value) const
    122 {
    123     if (!value.isFinite())
    124         return String();
    125     DateComponents date;
    126     if (!setMillisecondToDateComponents(value.toDouble(), &date))
    127         return String();
    128     return serializeWithComponents(date);
    129 }
    130 
    131 String BaseDateAndTimeInputType::serializeWithComponents(const DateComponents& date) const
    132 {
    133     Decimal step;
    134     if (!element().getAllowedValueStep(&step))
    135         return date.toString();
    136     if (step.remainder(msecPerMinute).isZero())
    137         return date.toString(DateComponents::None);
    138     if (step.remainder(msecPerSecond).isZero())
    139         return date.toString(DateComponents::Second);
    140     return date.toString(DateComponents::Millisecond);
    141 }
    142 
    143 String BaseDateAndTimeInputType::serializeWithMilliseconds(double value) const
    144 {
    145     return serialize(Decimal::fromDouble(value));
    146 }
    147 
    148 String BaseDateAndTimeInputType::localizeValue(const String& proposedValue) const
    149 {
    150     DateComponents date;
    151     if (!parseToDateComponents(proposedValue, &date))
    152         return proposedValue;
    153 
    154     String localized = element().locale().formatDateTime(date);
    155     return localized.isEmpty() ? proposedValue : localized;
    156 }
    157 
    158 String BaseDateAndTimeInputType::visibleValue() const
    159 {
    160     return localizeValue(element().value());
    161 }
    162 
    163 String BaseDateAndTimeInputType::sanitizeValue(const String& proposedValue) const
    164 {
    165     return typeMismatchFor(proposedValue) ? emptyString() : proposedValue;
    166 }
    167 
    168 bool BaseDateAndTimeInputType::supportsReadOnly() const
    169 {
    170     return true;
    171 }
    172 
    173 bool BaseDateAndTimeInputType::shouldRespectListAttribute()
    174 {
    175     return true;
    176 }
    177 
    178 bool BaseDateAndTimeInputType::valueMissing(const String& value) const
    179 {
    180     return element().isRequired() && value.isEmpty();
    181 }
    182 
    183 bool BaseDateAndTimeInputType::shouldShowFocusRingOnMouseFocus() const
    184 {
    185     return true;
    186 }
    187 
    188 } // namespace blink
    189