• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2012 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
6  * are met:
7  * 1.  Redistributions of source code must retain the above copyright
8  *     notice, this list of conditions and the following disclaimer.
9  * 2.  Redistributions in binary form must reproduce the above copyright
10  *     notice, this list of conditions and the following disclaimer in the
11  *     documentation and/or other materials provided with the distribution.
12  *
13  * THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS'' AND
14  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
16  * ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS BE LIABLE
17  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
18  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
19  * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
20  * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
21  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
22  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
23  * SUCH DAMAGE.
24  */
25 
26 #include "config.h"
27 #if ENABLE(INPUT_MULTIPLE_FIELDS_UI)
28 #include "core/html/shadow/DateTimeFieldElement.h"
29 
30 #include "HTMLNames.h"
31 #include "core/dom/Text.h"
32 #include "core/events/KeyboardEvent.h"
33 #include "platform/text/PlatformLocale.h"
34 #include "wtf/text/WTFString.h"
35 
36 namespace WebCore {
37 
38 using namespace HTMLNames;
39 
emptyValueAXText()40 static String emptyValueAXText()
41 {
42     return Locale::defaultLocale().queryString(blink::WebLocalizedString::AXDateTimeFieldEmptyValueText);
43 }
44 
~FieldOwner()45 DateTimeFieldElement::FieldOwner::~FieldOwner()
46 {
47 }
48 
DateTimeFieldElement(Document & document,FieldOwner & fieldOwner)49 DateTimeFieldElement::DateTimeFieldElement(Document& document, FieldOwner& fieldOwner)
50     : HTMLSpanElement(document)
51     , m_fieldOwner(&fieldOwner)
52 {
53 }
54 
defaultEventHandler(Event * event)55 void DateTimeFieldElement::defaultEventHandler(Event* event)
56 {
57     if (event->type() == EventTypeNames::blur)
58         didBlur();
59 
60     if (event->type() == EventTypeNames::focus)
61         didFocus();
62 
63     if (event->isKeyboardEvent()) {
64         KeyboardEvent* keyboardEvent = toKeyboardEvent(event);
65         if (!isDisabled() && !isFieldOwnerDisabled() && !isFieldOwnerReadOnly()) {
66             handleKeyboardEvent(keyboardEvent);
67             if (keyboardEvent->defaultHandled())
68                 return;
69         }
70         defaultKeyboardEventHandler(keyboardEvent);
71         if (keyboardEvent->defaultHandled())
72             return;
73     }
74 
75     HTMLElement::defaultEventHandler(event);
76 }
77 
defaultKeyboardEventHandler(KeyboardEvent * keyboardEvent)78 void DateTimeFieldElement::defaultKeyboardEventHandler(KeyboardEvent* keyboardEvent)
79 {
80     if (keyboardEvent->type() != EventTypeNames::keydown)
81         return;
82 
83     if (isDisabled() || isFieldOwnerDisabled())
84         return;
85 
86     const String& keyIdentifier = keyboardEvent->keyIdentifier();
87 
88     if (keyIdentifier == "Left") {
89         if (!m_fieldOwner)
90             return;
91         // FIXME: We'd like to use FocusController::advanceFocus(FocusDirectionLeft, ...)
92         // but it doesn't work for shadow nodes. webkit.org/b/104650
93         if (!localeForOwner().isRTL() && m_fieldOwner->focusOnPreviousField(*this))
94             keyboardEvent->setDefaultHandled();
95         return;
96     }
97 
98     if (keyIdentifier == "Right") {
99         if (!m_fieldOwner)
100             return;
101         // FIXME: We'd like to use FocusController::advanceFocus(FocusDirectionRight, ...)
102         // but it doesn't work for shadow nodes. webkit.org/b/104650
103         if (!localeForOwner().isRTL() && m_fieldOwner->focusOnNextField(*this))
104             keyboardEvent->setDefaultHandled();
105         return;
106     }
107 
108     if (isFieldOwnerReadOnly())
109         return;
110 
111     if (keyIdentifier == "Down") {
112         if (keyboardEvent->getModifierState("Alt"))
113             return;
114         keyboardEvent->setDefaultHandled();
115         stepDown();
116         return;
117     }
118 
119     if (keyIdentifier == "Up") {
120         keyboardEvent->setDefaultHandled();
121         stepUp();
122         return;
123     }
124 
125     if (keyIdentifier == "U+0008" || keyIdentifier == "U+007F") {
126         keyboardEvent->setDefaultHandled();
127         setEmptyValue(DispatchEvent);
128         return;
129     }
130 }
131 
didBlur()132 void DateTimeFieldElement::didBlur()
133 {
134     if (m_fieldOwner)
135         m_fieldOwner->didBlurFromField();
136 }
137 
didFocus()138 void DateTimeFieldElement::didFocus()
139 {
140     if (m_fieldOwner)
141         m_fieldOwner->didFocusOnField();
142 }
143 
focusOnNextField()144 void DateTimeFieldElement::focusOnNextField()
145 {
146     if (!m_fieldOwner)
147         return;
148     m_fieldOwner->focusOnNextField(*this);
149 }
150 
initialize(const AtomicString & pseudo,const String & axHelpText,int axMinimum,int axMaximum)151 void DateTimeFieldElement::initialize(const AtomicString& pseudo, const String& axHelpText, int axMinimum, int axMaximum)
152 {
153     // On accessibility, DateTimeFieldElement acts like spin button.
154     setAttribute(roleAttr, AtomicString("spinbutton", AtomicString::ConstructFromLiteral));
155     setAttribute(aria_valuetextAttr, emptyValueAXText());
156     setAttribute(aria_valueminAttr, String::number(axMinimum));
157     setAttribute(aria_valuemaxAttr, String::number(axMaximum));
158 
159     setAttribute(aria_helpAttr, axHelpText);
160     setPseudo(pseudo);
161     appendChild(Text::create(document(), visibleValue()));
162 }
163 
isDateTimeFieldElement() const164 bool DateTimeFieldElement::isDateTimeFieldElement() const
165 {
166     return true;
167 }
168 
isFieldOwnerDisabled() const169 bool DateTimeFieldElement::isFieldOwnerDisabled() const
170 {
171     return m_fieldOwner && m_fieldOwner->isFieldOwnerDisabled();
172 }
173 
isFieldOwnerReadOnly() const174 bool DateTimeFieldElement::isFieldOwnerReadOnly() const
175 {
176     return m_fieldOwner && m_fieldOwner->isFieldOwnerReadOnly();
177 }
178 
isDisabled() const179 bool DateTimeFieldElement::isDisabled() const
180 {
181     return fastHasAttribute(disabledAttr);
182 }
183 
localeForOwner() const184 Locale& DateTimeFieldElement::localeForOwner() const
185 {
186     return document().getCachedLocale(localeIdentifier());
187 }
188 
localeIdentifier() const189 AtomicString DateTimeFieldElement::localeIdentifier() const
190 {
191     return m_fieldOwner ? m_fieldOwner->localeIdentifier() : nullAtom;
192 }
193 
maximumWidth(const Font &)194 float DateTimeFieldElement::maximumWidth(const Font&)
195 {
196     const float paddingLeftAndRight = 2; // This should match to html.css.
197     return paddingLeftAndRight;
198 }
199 
setDisabled()200 void DateTimeFieldElement::setDisabled()
201 {
202     // Set HTML attribute disabled to change apperance.
203     setBooleanAttribute(disabledAttr, true);
204     setNeedsStyleRecalc();
205 }
206 
supportsFocus() const207 bool DateTimeFieldElement::supportsFocus() const
208 {
209     return !isDisabled() && !isFieldOwnerDisabled();
210 }
211 
updateVisibleValue(EventBehavior eventBehavior)212 void DateTimeFieldElement::updateVisibleValue(EventBehavior eventBehavior)
213 {
214     Text* const textNode = toText(firstChild());
215     const String newVisibleValue = visibleValue();
216     ASSERT(newVisibleValue.length() > 0);
217 
218     if (textNode->wholeText() == newVisibleValue)
219         return;
220 
221     textNode->replaceWholeText(newVisibleValue);
222     if (hasValue()) {
223         setAttribute(aria_valuetextAttr, newVisibleValue);
224         setAttribute(aria_valuenowAttr, String::number(valueForARIAValueNow()));
225     } else {
226         setAttribute(aria_valuetextAttr, emptyValueAXText());
227         removeAttribute(aria_valuenowAttr);
228     }
229 
230     if (eventBehavior == DispatchEvent && m_fieldOwner)
231         m_fieldOwner->fieldValueChanged();
232 }
233 
valueForARIAValueNow() const234 int DateTimeFieldElement::valueForARIAValueNow() const
235 {
236     return valueAsInteger();
237 }
238 
239 } // namespace WebCore
240 
241 #endif
242