HTMLFormControlElement.cpp [plain text]
#include "config.h"
#include "HTMLFormControlElement.h"
#include "Chrome.h"
#include "ChromeClient.h"
#include "Document.h"
#include "Event.h"
#include "EventHandler.h"
#include "EventNames.h"
#include "Frame.h"
#include "HTMLFormElement.h"
#include "HTMLInputElement.h"
#include "HTMLNames.h"
#include "HTMLParser.h"
#include "HTMLTokenizer.h"
#include "MappedAttribute.h"
#include "Page.h"
#include "RenderBox.h"
#include "RenderTextControl.h"
#include "RenderTheme.h"
#include "ScriptEventListener.h"
#include "ValidityState.h"
namespace WebCore {
using namespace HTMLNames;
HTMLFormControlElement::HTMLFormControlElement(const QualifiedName& tagName, Document* doc, HTMLFormElement* f, ConstructionType constructionType)
: HTMLElement(tagName, doc, constructionType)
, m_form(f)
, m_disabled(false)
, m_readOnly(false)
, m_required(false)
, m_valueMatchesRenderer(false)
, m_willValidateInitialized(false)
, m_willValidate(true)
, m_isValid(true)
{
if (!m_form)
m_form = findFormAncestor();
if (m_form)
m_form->registerFormElement(this);
}
HTMLFormControlElement::~HTMLFormControlElement()
{
if (m_form)
m_form->removeFormElement(this);
}
bool HTMLFormControlElement::formNoValidate() const
{
return !getAttribute(formnovalidateAttr).isNull();
}
void HTMLFormControlElement::setFormNoValidate(bool formnovalidate)
{
setAttribute(formnovalidateAttr, formnovalidate ? "" : 0);
}
ValidityState* HTMLFormControlElement::validity()
{
if (!m_validityState)
m_validityState = ValidityState::create(this);
return m_validityState.get();
}
void HTMLFormControlElement::parseMappedAttribute(MappedAttribute *attr)
{
if (attr->name() == disabledAttr) {
bool oldDisabled = m_disabled;
m_disabled = !attr->isNull();
if (oldDisabled != m_disabled) {
setNeedsStyleRecalc();
if (renderer() && renderer()->style()->hasAppearance())
renderer()->theme()->stateChanged(renderer(), EnabledState);
}
} else if (attr->name() == readonlyAttr) {
bool oldReadOnly = m_readOnly;
m_readOnly = !attr->isNull();
if (oldReadOnly != m_readOnly) {
setNeedsStyleRecalc();
if (renderer() && renderer()->style()->hasAppearance())
renderer()->theme()->stateChanged(renderer(), ReadOnlyState);
}
} else if (attr->name() == requiredAttr) {
bool oldRequired = m_required;
m_required = !attr->isNull();
if (oldRequired != m_required) {
setNeedsValidityCheck();
setNeedsStyleRecalc(); }
} else
HTMLElement::parseMappedAttribute(attr);
setNeedsWillValidateCheck();
}
void HTMLFormControlElement::attach()
{
ASSERT(!attached());
HTMLElement::attach();
if (renderer())
renderer()->updateFromElement();
bool isInputTypeHidden = false;
if (hasTagName(inputTag))
isInputTypeHidden = static_cast<HTMLInputElement*>(this)->isInputTypeHidden();
if (autofocus() && renderer() && !document()->ignoreAutofocus() && !isReadOnlyFormControl() &&
((hasTagName(inputTag) && !isInputTypeHidden) || hasTagName(selectTag) ||
hasTagName(buttonTag) || hasTagName(textareaTag)))
focus();
}
void HTMLFormControlElement::insertedIntoTree(bool deep)
{
if (!m_form) {
m_form = findFormAncestor();
if (m_form)
m_form->registerFormElement(this);
else
document()->checkedRadioButtons().addButton(this);
}
HTMLElement::insertedIntoTree(deep);
}
static inline Node* findRoot(Node* n)
{
Node* root = n;
for (; n; n = n->parentNode())
root = n;
return root;
}
void HTMLFormControlElement::removedFromTree(bool deep)
{
HTMLParser* parser = 0;
if (Tokenizer* tokenizer = document()->tokenizer())
if (tokenizer->isHTMLTokenizer())
parser = static_cast<HTMLTokenizer*>(tokenizer)->htmlParser();
if (m_form && !(parser && parser->isHandlingResidualStyleAcrossBlocks()) && findRoot(this) != findRoot(m_form)) {
m_form->removeFormElement(this);
m_form = 0;
}
HTMLElement::removedFromTree(deep);
}
const AtomicString& HTMLFormControlElement::formControlName() const
{
const AtomicString& name = fastGetAttribute(nameAttr);
return name.isNull() ? emptyAtom : name;
}
void HTMLFormControlElement::setName(const AtomicString &value)
{
setAttribute(nameAttr, value);
}
void HTMLFormControlElement::dispatchFormControlChangeEvent()
{
dispatchEvent(Event::create(eventNames().changeEvent, true, false));
}
void HTMLFormControlElement::setDisabled(bool b)
{
setAttribute(disabledAttr, b ? "" : 0);
}
void HTMLFormControlElement::setReadOnly(bool b)
{
setAttribute(readonlyAttr, b ? "" : 0);
}
bool HTMLFormControlElement::autofocus() const
{
return hasAttribute(autofocusAttr);
}
void HTMLFormControlElement::setAutofocus(bool b)
{
setAttribute(autofocusAttr, b ? "autofocus" : 0);
}
bool HTMLFormControlElement::required() const
{
return m_required;
}
void HTMLFormControlElement::setRequired(bool b)
{
setAttribute(requiredAttr, b ? "required" : 0);
}
static void updateFromElementCallback(Node* node)
{
ASSERT_ARG(node, node->isElementNode());
ASSERT_ARG(node, static_cast<Element*>(node)->isFormControlElement());
ASSERT(node->renderer());
if (RenderObject* renderer = node->renderer())
renderer->updateFromElement();
}
void HTMLFormControlElement::recalcStyle(StyleChange change)
{
HTMLElement::recalcStyle(change);
if (renderer())
queuePostAttachCallback(updateFromElementCallback, this);
}
bool HTMLFormControlElement::supportsFocus() const
{
return !disabled();
}
bool HTMLFormControlElement::isFocusable() const
{
if (!renderer() ||
!renderer()->isBox() || toRenderBox(renderer())->size().isEmpty())
return false;
return HTMLElement::isFocusable();
}
bool HTMLFormControlElement::isKeyboardFocusable(KeyboardEvent* event) const
{
if (isFocusable())
if (document()->frame())
return document()->frame()->eventHandler()->tabsToAllControls(event);
return false;
}
bool HTMLFormControlElement::isMouseFocusable() const
{
#if PLATFORM(GTK)
return HTMLElement::isMouseFocusable();
#else
return false;
#endif
}
short HTMLFormControlElement::tabIndex() const
{
return Element::tabIndex();
}
bool HTMLFormControlElement::recalcWillValidate() const
{
return !m_disabled && !m_readOnly;
}
bool HTMLFormControlElement::willValidate() const
{
if (!m_willValidateInitialized) {
m_willValidateInitialized = true;
m_willValidate = recalcWillValidate();
} else {
ASSERT(m_willValidate == recalcWillValidate());
}
return m_willValidate;
}
void HTMLFormControlElement::setNeedsWillValidateCheck()
{
bool newWillValidate = recalcWillValidate();
if (m_willValidateInitialized && m_willValidate == newWillValidate)
return;
m_willValidateInitialized = true;
m_willValidate = newWillValidate;
setNeedsStyleRecalc();
}
String HTMLFormControlElement::validationMessage()
{
return validity()->validationMessage();
}
bool HTMLFormControlElement::checkValidity(Vector<RefPtr<HTMLFormControlElement> >* unhandledInvalidControls)
{
if (!willValidate() || isValidFormControlElement())
return true;
RefPtr<HTMLFormControlElement> protector(this);
RefPtr<Document> originalDocument(document());
bool needsDefaultAction = dispatchEvent(Event::create(eventNames().invalidEvent, false, true));
if (needsDefaultAction && unhandledInvalidControls && inDocument() && originalDocument == document())
unhandledInvalidControls->append(this);
return false;
}
bool HTMLFormControlElement::isValidFormControlElement()
{
ASSERT(m_isValid == validity()->valid());
return m_isValid;
}
void HTMLFormControlElement::setNeedsValidityCheck()
{
bool newIsValid = validity()->valid();
if (willValidate() && newIsValid != m_isValid) {
setNeedsStyleRecalc();
}
m_isValid = newIsValid;
}
void HTMLFormControlElement::setCustomValidity(const String& error)
{
validity()->setCustomErrorMessage(error);
}
void HTMLFormControlElement::dispatchFocusEvent()
{
if (document()->page())
document()->page()->chrome()->client()->formDidFocus(this);
HTMLElement::dispatchFocusEvent();
}
void HTMLFormControlElement::dispatchBlurEvent()
{
if (document()->page())
document()->page()->chrome()->client()->formDidBlur(this);
HTMLElement::dispatchBlurEvent();
}
HTMLFormElement* HTMLFormControlElement::virtualForm() const
{
return m_form;
}
bool HTMLFormControlElement::isDefaultButtonForForm() const
{
return isSuccessfulSubmitButton() && m_form && m_form->defaultButton() == this;
}
void HTMLFormControlElement::removeFromForm()
{
if (!m_form)
return;
m_form->removeFormElement(this);
m_form = 0;
}
HTMLFormControlElementWithState::HTMLFormControlElementWithState(const QualifiedName& tagName, Document* doc, HTMLFormElement* f)
: HTMLFormControlElement(tagName, doc, f)
{
document()->registerFormElementWithState(this);
}
HTMLFormControlElementWithState::~HTMLFormControlElementWithState()
{
document()->unregisterFormElementWithState(this);
}
void HTMLFormControlElementWithState::willMoveToNewOwnerDocument()
{
document()->unregisterFormElementWithState(this);
HTMLFormControlElement::willMoveToNewOwnerDocument();
}
void HTMLFormControlElementWithState::didMoveToNewOwnerDocument()
{
document()->registerFormElementWithState(this);
HTMLFormControlElement::didMoveToNewOwnerDocument();
}
bool HTMLFormControlElementWithState::autoComplete() const
{
if (!form())
return true;
return form()->autoComplete();
}
bool HTMLFormControlElementWithState::shouldSaveAndRestoreFormControlState() const
{
return autoComplete();
}
void HTMLFormControlElementWithState::finishParsingChildren()
{
HTMLFormControlElement::finishParsingChildren();
if (!shouldSaveAndRestoreFormControlState())
return;
Document* doc = document();
if (doc->hasStateForNewFormElements()) {
String state;
if (doc->takeStateForFormElement(name().impl(), type().impl(), state))
restoreFormControlState(state);
}
}
void HTMLFormControlElementWithState::defaultEventHandler(Event* event)
{
if (event->type() == eventNames().webkitEditableContentChangedEvent && renderer() && renderer()->isTextControl()) {
toRenderTextControl(renderer())->subtreeHasChanged();
return;
}
HTMLFormControlElement::defaultEventHandler(event);
}
HTMLTextFormControlElement::HTMLTextFormControlElement(const QualifiedName& tagName, Document* doc, HTMLFormElement* form)
: HTMLFormControlElementWithState(tagName, doc, form)
{
}
HTMLTextFormControlElement::~HTMLTextFormControlElement()
{
}
void HTMLTextFormControlElement::dispatchFocusEvent()
{
if (supportsPlaceholder())
updatePlaceholderVisibility(false);
handleFocusEvent();
HTMLFormControlElementWithState::dispatchFocusEvent();
}
void HTMLTextFormControlElement::dispatchBlurEvent()
{
if (supportsPlaceholder())
updatePlaceholderVisibility(false);
handleBlurEvent();
HTMLFormControlElementWithState::dispatchBlurEvent();
}
bool HTMLTextFormControlElement::placeholderShouldBeVisible() const
{
return supportsPlaceholder()
&& isEmptyValue()
&& document()->focusedNode() != this
&& !getAttribute(placeholderAttr).isEmpty();
}
void HTMLTextFormControlElement::updatePlaceholderVisibility(bool placeholderValueChanged)
{
if (supportsPlaceholder() && renderer())
toRenderTextControl(renderer())->updatePlaceholderVisibility(placeholderShouldBeVisible(), placeholderValueChanged);
}
RenderTextControl* HTMLTextFormControlElement::textRendererAfterUpdateLayout()
{
if (!isTextFormControl())
return 0;
document()->updateLayoutIgnorePendingStylesheets();
return toRenderTextControl(renderer());
}
void HTMLTextFormControlElement::setSelectionStart(int start)
{
if (RenderTextControl* renderer = textRendererAfterUpdateLayout())
renderer->setSelectionStart(start);
}
void HTMLTextFormControlElement::setSelectionEnd(int end)
{
if (RenderTextControl* renderer = textRendererAfterUpdateLayout())
renderer->setSelectionEnd(end);
}
void HTMLTextFormControlElement::select()
{
if (RenderTextControl* renderer = textRendererAfterUpdateLayout())
renderer->select();
}
void HTMLTextFormControlElement::setSelectionRange(int start, int end)
{
if (RenderTextControl* renderer = textRendererAfterUpdateLayout())
renderer->setSelectionRange(start, end);
}
int HTMLTextFormControlElement::selectionStart()
{
if (!isTextFormControl())
return 0;
if (document()->focusedNode() != this && cachedSelectionStart() >= 0)
return cachedSelectionStart();
if (!renderer())
return 0;
return toRenderTextControl(renderer())->selectionStart();
}
int HTMLTextFormControlElement::selectionEnd()
{
if (!isTextFormControl())
return 0;
if (document()->focusedNode() != this && cachedSelectionEnd() >= 0)
return cachedSelectionEnd();
if (!renderer())
return 0;
return toRenderTextControl(renderer())->selectionEnd();
}
VisibleSelection HTMLTextFormControlElement::selection() const
{
if (!renderer() || !isTextFormControl() || cachedSelectionStart() < 0 || cachedSelectionEnd() < 0)
return VisibleSelection();
return toRenderTextControl(renderer())->selection(cachedSelectionStart(), cachedSelectionEnd());
}
void HTMLTextFormControlElement::parseMappedAttribute(MappedAttribute* attr)
{
if (attr->name() == placeholderAttr)
updatePlaceholderVisibility(true);
else if (attr->name() == onselectAttr)
setAttributeEventListener(eventNames().selectEvent, createAttributeEventListener(this, attr));
else if (attr->name() == onchangeAttr)
setAttributeEventListener(eventNames().changeEvent, createAttributeEventListener(this, attr));
else
HTMLFormControlElementWithState::parseMappedAttribute(attr);
}
}