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