2015-05-03 22:32:37 +03:00
|
|
|
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
|
|
/* vim: set ts=8 sts=2 et sw=2 tw=80: */
|
2012-05-21 15:12:37 +04:00
|
|
|
/* 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/. */
|
2011-08-08 21:31:32 +04:00
|
|
|
|
2013-08-14 10:55:46 +04:00
|
|
|
#include "mozilla/dom/HTMLMenuItemElement.h"
|
|
|
|
|
2013-09-25 15:21:22 +04:00
|
|
|
#include "mozilla/BasicEvents.h"
|
2014-03-18 08:48:21 +04:00
|
|
|
#include "mozilla/EventDispatcher.h"
|
2013-02-18 14:06:27 +04:00
|
|
|
#include "mozilla/dom/HTMLMenuItemElementBinding.h"
|
2012-09-30 20:40:24 +04:00
|
|
|
#include "nsAttrValueInlines.h"
|
2013-08-14 10:55:46 +04:00
|
|
|
#include "nsContentUtils.h"
|
2011-08-08 21:31:32 +04:00
|
|
|
|
2013-02-18 14:05:52 +04:00
|
|
|
NS_IMPL_NS_NEW_HTML_ELEMENT_CHECK_PARSER(MenuItem)
|
|
|
|
|
|
|
|
namespace mozilla {
|
|
|
|
namespace dom {
|
2011-08-08 21:31:32 +04:00
|
|
|
|
|
|
|
// First bits are needed for the menuitem type.
|
|
|
|
#define NS_CHECKED_IS_TOGGLED (1 << 2)
|
|
|
|
#define NS_ORIGINAL_CHECKED_VALUE (1 << 3)
|
|
|
|
#define NS_MENUITEM_TYPE(bits) \
|
|
|
|
((bits) & ~(NS_CHECKED_IS_TOGGLED | NS_ORIGINAL_CHECKED_VALUE))
|
|
|
|
|
2016-09-07 05:20:17 +03:00
|
|
|
enum CmdType : uint8_t {
|
2011-08-08 21:31:32 +04:00
|
|
|
CMD_TYPE_MENUITEM = 1,
|
|
|
|
CMD_TYPE_CHECKBOX,
|
|
|
|
CMD_TYPE_RADIO
|
|
|
|
};
|
|
|
|
|
|
|
|
static const nsAttrValue::EnumTable kMenuItemTypeTable[] = {
|
|
|
|
{"menuitem", CMD_TYPE_MENUITEM},
|
|
|
|
{"checkbox", CMD_TYPE_CHECKBOX},
|
|
|
|
{"radio", CMD_TYPE_RADIO},
|
2016-09-07 05:20:17 +03:00
|
|
|
{nullptr, 0}};
|
2011-08-08 21:31:32 +04:00
|
|
|
|
|
|
|
static const nsAttrValue::EnumTable* kMenuItemDefaultType =
|
|
|
|
&kMenuItemTypeTable[0];
|
|
|
|
|
|
|
|
// A base class inherited by all radio visitors.
|
|
|
|
class Visitor {
|
|
|
|
public:
|
|
|
|
Visitor() {}
|
|
|
|
virtual ~Visitor() {}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Visit a node in the tree. This is meant to be called on all radios in a
|
|
|
|
* group, sequentially. If the method returns false then the iteration is
|
|
|
|
* stopped.
|
|
|
|
*/
|
2013-02-18 14:05:52 +04:00
|
|
|
virtual bool Visit(HTMLMenuItemElement* aMenuItem) = 0;
|
2011-08-08 21:31:32 +04:00
|
|
|
};
|
|
|
|
|
|
|
|
// Find the selected radio, see GetSelectedRadio().
|
|
|
|
class GetCheckedVisitor : public Visitor {
|
|
|
|
public:
|
2014-09-02 04:49:25 +04:00
|
|
|
explicit GetCheckedVisitor(HTMLMenuItemElement** aResult)
|
2011-08-08 21:31:32 +04:00
|
|
|
: mResult(aResult) {}
|
2017-11-06 06:37:28 +03:00
|
|
|
virtual bool Visit(HTMLMenuItemElement* aMenuItem) override {
|
2011-08-08 21:31:32 +04:00
|
|
|
if (aMenuItem->IsChecked()) {
|
|
|
|
*mResult = aMenuItem;
|
2011-10-17 18:59:28 +04:00
|
|
|
return false;
|
2011-08-08 21:31:32 +04:00
|
|
|
}
|
2011-10-17 18:59:28 +04:00
|
|
|
return true;
|
2011-08-08 21:31:32 +04:00
|
|
|
}
|
2018-11-30 13:46:48 +03:00
|
|
|
|
2011-08-08 21:31:32 +04:00
|
|
|
protected:
|
2013-02-18 14:05:52 +04:00
|
|
|
HTMLMenuItemElement** mResult;
|
2011-08-08 21:31:32 +04:00
|
|
|
};
|
|
|
|
|
|
|
|
// Deselect all radios except the one passed to the constructor.
|
|
|
|
class ClearCheckedVisitor : public Visitor {
|
|
|
|
public:
|
2014-09-02 04:49:25 +04:00
|
|
|
explicit ClearCheckedVisitor(HTMLMenuItemElement* aExcludeMenuItem)
|
2011-08-08 21:31:32 +04:00
|
|
|
: mExcludeMenuItem(aExcludeMenuItem) {}
|
2017-11-06 06:37:28 +03:00
|
|
|
virtual bool Visit(HTMLMenuItemElement* aMenuItem) override {
|
2011-08-08 21:31:32 +04:00
|
|
|
if (aMenuItem != mExcludeMenuItem && aMenuItem->IsChecked()) {
|
|
|
|
aMenuItem->ClearChecked();
|
|
|
|
}
|
2011-10-17 18:59:28 +04:00
|
|
|
return true;
|
2011-08-08 21:31:32 +04:00
|
|
|
}
|
2018-11-30 13:46:48 +03:00
|
|
|
|
2011-08-08 21:31:32 +04:00
|
|
|
protected:
|
2013-02-18 14:05:52 +04:00
|
|
|
HTMLMenuItemElement* mExcludeMenuItem;
|
2011-08-08 21:31:32 +04:00
|
|
|
};
|
|
|
|
|
|
|
|
// Get current value of the checked dirty flag. The same value is stored on all
|
|
|
|
// radios in the group, so we need to check only the first one.
|
|
|
|
class GetCheckedDirtyVisitor : public Visitor {
|
|
|
|
public:
|
2011-09-29 10:19:26 +04:00
|
|
|
GetCheckedDirtyVisitor(bool* aCheckedDirty,
|
2013-02-18 14:05:52 +04:00
|
|
|
HTMLMenuItemElement* aExcludeMenuItem)
|
2011-08-08 21:31:32 +04:00
|
|
|
: mCheckedDirty(aCheckedDirty), mExcludeMenuItem(aExcludeMenuItem) {}
|
2017-11-06 06:37:28 +03:00
|
|
|
virtual bool Visit(HTMLMenuItemElement* aMenuItem) override {
|
2011-08-08 21:31:32 +04:00
|
|
|
if (aMenuItem == mExcludeMenuItem) {
|
2011-10-17 18:59:28 +04:00
|
|
|
return true;
|
2011-08-08 21:31:32 +04:00
|
|
|
}
|
|
|
|
*mCheckedDirty = aMenuItem->IsCheckedDirty();
|
2011-10-17 18:59:28 +04:00
|
|
|
return false;
|
2011-08-08 21:31:32 +04:00
|
|
|
}
|
2018-11-30 13:46:48 +03:00
|
|
|
|
2011-08-08 21:31:32 +04:00
|
|
|
protected:
|
2011-09-29 10:19:26 +04:00
|
|
|
bool* mCheckedDirty;
|
2013-02-18 14:05:52 +04:00
|
|
|
HTMLMenuItemElement* mExcludeMenuItem;
|
2011-08-08 21:31:32 +04:00
|
|
|
};
|
|
|
|
|
|
|
|
// Set checked dirty to true on all radios in the group.
|
|
|
|
class SetCheckedDirtyVisitor : public Visitor {
|
|
|
|
public:
|
|
|
|
SetCheckedDirtyVisitor() {}
|
2017-11-06 06:37:28 +03:00
|
|
|
virtual bool Visit(HTMLMenuItemElement* aMenuItem) override {
|
2011-08-08 21:31:32 +04:00
|
|
|
aMenuItem->SetCheckedDirty();
|
2011-10-17 18:59:28 +04:00
|
|
|
return true;
|
2011-08-08 21:31:32 +04:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
// A helper visitor that is used to combine two operations (visitors) to avoid
|
|
|
|
// iterating over radios twice.
|
|
|
|
class CombinedVisitor : public Visitor {
|
|
|
|
public:
|
|
|
|
CombinedVisitor(Visitor* aVisitor1, Visitor* aVisitor2)
|
|
|
|
: mVisitor1(aVisitor1),
|
|
|
|
mVisitor2(aVisitor2),
|
2011-10-17 18:59:28 +04:00
|
|
|
mContinue1(true),
|
|
|
|
mContinue2(true) {}
|
2017-11-06 06:37:28 +03:00
|
|
|
virtual bool Visit(HTMLMenuItemElement* aMenuItem) override {
|
2011-08-08 21:31:32 +04:00
|
|
|
if (mContinue1) {
|
|
|
|
mContinue1 = mVisitor1->Visit(aMenuItem);
|
|
|
|
}
|
|
|
|
if (mContinue2) {
|
|
|
|
mContinue2 = mVisitor2->Visit(aMenuItem);
|
|
|
|
}
|
|
|
|
return mContinue1 || mContinue2;
|
|
|
|
}
|
2018-11-30 13:46:48 +03:00
|
|
|
|
2011-08-08 21:31:32 +04:00
|
|
|
protected:
|
|
|
|
Visitor* mVisitor1;
|
|
|
|
Visitor* mVisitor2;
|
2011-09-29 10:19:26 +04:00
|
|
|
bool mContinue1;
|
|
|
|
bool mContinue2;
|
2011-08-08 21:31:32 +04:00
|
|
|
};
|
|
|
|
|
2013-02-18 14:05:52 +04:00
|
|
|
HTMLMenuItemElement::HTMLMenuItemElement(
|
2018-09-21 23:45:49 +03:00
|
|
|
already_AddRefed<mozilla::dom::NodeInfo>&& aNodeInfo,
|
|
|
|
FromParser aFromParser)
|
|
|
|
: nsGenericHTMLElement(std::move(aNodeInfo)),
|
2011-08-08 21:31:32 +04:00
|
|
|
mType(kMenuItemDefaultType->value),
|
|
|
|
mParserCreating(false),
|
|
|
|
mShouldInitChecked(false),
|
|
|
|
mCheckedDirty(false),
|
|
|
|
mChecked(false) {
|
|
|
|
mParserCreating = aFromParser;
|
|
|
|
}
|
|
|
|
|
2013-02-18 14:05:52 +04:00
|
|
|
HTMLMenuItemElement::~HTMLMenuItemElement() {}
|
2011-08-08 21:31:32 +04:00
|
|
|
|
2013-02-18 14:05:52 +04:00
|
|
|
// NS_IMPL_ELEMENT_CLONE(HTMLMenuItemElement)
|
2011-08-08 21:31:32 +04:00
|
|
|
|
2018-08-09 02:58:44 +03:00
|
|
|
nsresult HTMLMenuItemElement::Clone(dom::NodeInfo* aNodeInfo,
|
|
|
|
nsINode** aResult) const {
|
2012-07-30 18:20:58 +04:00
|
|
|
*aResult = nullptr;
|
2015-10-18 08:24:48 +03:00
|
|
|
RefPtr<HTMLMenuItemElement> it =
|
2018-09-21 23:45:49 +03:00
|
|
|
new HTMLMenuItemElement(do_AddRef(aNodeInfo), NOT_FROM_PARSER);
|
2018-08-09 02:58:44 +03:00
|
|
|
nsresult rv = const_cast<HTMLMenuItemElement*>(this)->CopyInnerTo(it);
|
2011-08-08 21:31:32 +04:00
|
|
|
if (NS_SUCCEEDED(rv)) {
|
|
|
|
switch (mType) {
|
|
|
|
case CMD_TYPE_CHECKBOX:
|
|
|
|
case CMD_TYPE_RADIO:
|
|
|
|
if (mCheckedDirty) {
|
|
|
|
// We no longer have our original checked state. Set our
|
|
|
|
// checked state on the clone.
|
|
|
|
it->mCheckedDirty = true;
|
|
|
|
it->mChecked = mChecked;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2011-08-11 17:29:50 +04:00
|
|
|
it.forget(aResult);
|
2011-08-08 21:31:32 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
2017-10-13 01:10:50 +03:00
|
|
|
void HTMLMenuItemElement::GetType(DOMString& aValue) {
|
|
|
|
GetEnumAttr(nsGkAtoms::type, kMenuItemDefaultType->tag, aValue);
|
2011-08-08 21:31:32 +04:00
|
|
|
}
|
|
|
|
|
2013-02-18 14:05:52 +04:00
|
|
|
void HTMLMenuItemElement::SetChecked(bool aChecked) {
|
2011-09-29 10:19:26 +04:00
|
|
|
bool checkedChanged = mChecked != aChecked;
|
2011-08-08 21:31:32 +04:00
|
|
|
|
|
|
|
mChecked = aChecked;
|
|
|
|
|
|
|
|
if (mType == CMD_TYPE_RADIO) {
|
|
|
|
if (checkedChanged) {
|
|
|
|
if (mCheckedDirty) {
|
|
|
|
ClearCheckedVisitor visitor(this);
|
|
|
|
WalkRadioGroup(&visitor);
|
|
|
|
} else {
|
|
|
|
ClearCheckedVisitor visitor1(this);
|
|
|
|
SetCheckedDirtyVisitor visitor2;
|
|
|
|
CombinedVisitor visitor(&visitor1, &visitor2);
|
|
|
|
WalkRadioGroup(&visitor);
|
|
|
|
}
|
|
|
|
} else if (!mCheckedDirty) {
|
|
|
|
SetCheckedDirtyVisitor visitor;
|
|
|
|
WalkRadioGroup(&visitor);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
mCheckedDirty = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-10-21 05:11:07 +03:00
|
|
|
void HTMLMenuItemElement::GetEventTargetParent(EventChainPreVisitor& aVisitor) {
|
2015-08-29 02:58:32 +03:00
|
|
|
if (aVisitor.mEvent->mMessage == eMouseClick) {
|
2011-09-29 10:19:26 +04:00
|
|
|
bool originalCheckedValue = false;
|
2011-08-08 21:31:32 +04:00
|
|
|
switch (mType) {
|
|
|
|
case CMD_TYPE_CHECKBOX:
|
|
|
|
originalCheckedValue = mChecked;
|
|
|
|
SetChecked(!originalCheckedValue);
|
|
|
|
aVisitor.mItemFlags |= NS_CHECKED_IS_TOGGLED;
|
|
|
|
break;
|
|
|
|
case CMD_TYPE_RADIO:
|
2017-10-13 01:10:50 +03:00
|
|
|
// casting back to Element* here to resolve nsISupports ambiguity.
|
|
|
|
Element* supports = GetSelectedRadio();
|
|
|
|
aVisitor.mItemData = supports;
|
2011-08-08 21:31:32 +04:00
|
|
|
|
|
|
|
originalCheckedValue = mChecked;
|
|
|
|
if (!originalCheckedValue) {
|
2011-10-17 18:59:28 +04:00
|
|
|
SetChecked(true);
|
2011-08-08 21:31:32 +04:00
|
|
|
aVisitor.mItemFlags |= NS_CHECKED_IS_TOGGLED;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (originalCheckedValue) {
|
|
|
|
aVisitor.mItemFlags |= NS_ORIGINAL_CHECKED_VALUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
// We must cache type because mType may change during JS event.
|
|
|
|
aVisitor.mItemFlags |= mType;
|
|
|
|
}
|
|
|
|
|
2018-04-05 20:42:41 +03:00
|
|
|
nsGenericHTMLElement::GetEventTargetParent(aVisitor);
|
2011-08-08 21:31:32 +04:00
|
|
|
}
|
|
|
|
|
2014-03-18 08:48:20 +04:00
|
|
|
nsresult HTMLMenuItemElement::PostHandleEvent(EventChainPostVisitor& aVisitor) {
|
2011-08-08 21:31:32 +04:00
|
|
|
// Check to see if the event was cancelled.
|
2015-08-29 02:58:32 +03:00
|
|
|
if (aVisitor.mEvent->mMessage == eMouseClick &&
|
2011-08-08 21:31:32 +04:00
|
|
|
aVisitor.mItemFlags & NS_CHECKED_IS_TOGGLED &&
|
|
|
|
aVisitor.mEventStatus == nsEventStatus_eConsumeNoDefault) {
|
2011-09-29 10:19:26 +04:00
|
|
|
bool originalCheckedValue =
|
2011-08-08 21:31:32 +04:00
|
|
|
!!(aVisitor.mItemFlags & NS_ORIGINAL_CHECKED_VALUE);
|
2012-08-22 19:56:38 +04:00
|
|
|
uint8_t oldType = NS_MENUITEM_TYPE(aVisitor.mItemFlags);
|
2011-08-08 21:31:32 +04:00
|
|
|
|
2017-10-13 01:10:50 +03:00
|
|
|
nsCOMPtr<nsIContent> content(do_QueryInterface(aVisitor.mItemData));
|
2018-03-22 00:39:04 +03:00
|
|
|
RefPtr<HTMLMenuItemElement> selectedRadio =
|
|
|
|
HTMLMenuItemElement::FromNodeOrNull(content);
|
2011-08-08 21:31:32 +04:00
|
|
|
if (selectedRadio) {
|
2011-10-17 18:59:28 +04:00
|
|
|
selectedRadio->SetChecked(true);
|
2011-08-08 21:31:32 +04:00
|
|
|
if (mType != CMD_TYPE_RADIO) {
|
2011-10-17 18:59:28 +04:00
|
|
|
SetChecked(false);
|
2011-08-08 21:31:32 +04:00
|
|
|
}
|
|
|
|
} else if (oldType == CMD_TYPE_CHECKBOX) {
|
|
|
|
SetChecked(originalCheckedValue);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2019-01-02 16:05:23 +03:00
|
|
|
nsresult HTMLMenuItemElement::BindToTree(Document* aDocument,
|
2013-02-18 14:05:52 +04:00
|
|
|
nsIContent* aParent,
|
2018-07-31 21:18:38 +03:00
|
|
|
nsIContent* aBindingParent) {
|
2011-08-08 21:31:32 +04:00
|
|
|
nsresult rv =
|
|
|
|
nsGenericHTMLElement::BindToTree(aDocument, aParent, aBindingParent);
|
|
|
|
|
|
|
|
if (NS_SUCCEEDED(rv) && aDocument && mType == CMD_TYPE_RADIO) {
|
|
|
|
AddedToRadioGroup();
|
|
|
|
}
|
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
2013-02-18 14:05:52 +04:00
|
|
|
bool HTMLMenuItemElement::ParseAttribute(int32_t aNamespaceID,
|
2017-10-03 01:05:19 +03:00
|
|
|
nsAtom* aAttribute,
|
2013-02-18 14:05:52 +04:00
|
|
|
const nsAString& aValue,
|
2017-11-02 06:35:52 +03:00
|
|
|
nsIPrincipal* aMaybeScriptedPrincipal,
|
2013-02-18 14:05:52 +04:00
|
|
|
nsAttrValue& aResult) {
|
2011-08-08 21:31:32 +04:00
|
|
|
if (aNamespaceID == kNameSpaceID_None) {
|
|
|
|
if (aAttribute == nsGkAtoms::type) {
|
2017-04-20 07:35:20 +03:00
|
|
|
return aResult.ParseEnumValue(aValue, kMenuItemTypeTable, false,
|
|
|
|
kMenuItemDefaultType);
|
2011-08-08 21:31:32 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
if (aAttribute == nsGkAtoms::radiogroup) {
|
|
|
|
aResult.ParseAtom(aValue);
|
2011-10-17 18:59:28 +04:00
|
|
|
return true;
|
2011-08-08 21:31:32 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nsGenericHTMLElement::ParseAttribute(aNamespaceID, aAttribute, aValue,
|
2017-11-02 06:35:52 +03:00
|
|
|
aMaybeScriptedPrincipal, aResult);
|
2011-08-08 21:31:32 +04:00
|
|
|
}
|
|
|
|
|
2013-02-18 14:05:52 +04:00
|
|
|
void HTMLMenuItemElement::DoneCreatingElement() {
|
2011-08-08 21:31:32 +04:00
|
|
|
mParserCreating = false;
|
|
|
|
|
|
|
|
if (mShouldInitChecked) {
|
|
|
|
InitChecked();
|
|
|
|
mShouldInitChecked = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-02-18 14:05:52 +04:00
|
|
|
void HTMLMenuItemElement::GetText(nsAString& aText) {
|
2011-08-11 10:07:26 +04:00
|
|
|
nsAutoString text;
|
2015-05-22 21:16:20 +03:00
|
|
|
nsContentUtils::GetNodeTextContent(this, false, text);
|
2011-08-11 10:07:26 +04:00
|
|
|
|
2011-10-17 18:59:28 +04:00
|
|
|
text.CompressWhitespace(true, true);
|
2011-08-11 10:07:26 +04:00
|
|
|
aText = text;
|
|
|
|
}
|
|
|
|
|
2017-10-03 01:05:19 +03:00
|
|
|
nsresult HTMLMenuItemElement::AfterSetAttr(int32_t aNameSpaceID, nsAtom* aName,
|
2017-05-19 00:09:01 +03:00
|
|
|
const nsAttrValue* aValue,
|
2017-10-10 00:33:38 +03:00
|
|
|
const nsAttrValue* aOldValue,
|
|
|
|
nsIPrincipal* aSubjectPrincipal,
|
|
|
|
bool aNotify) {
|
2011-08-08 21:31:32 +04:00
|
|
|
if (aNameSpaceID == kNameSpaceID_None) {
|
2017-04-20 07:35:20 +03:00
|
|
|
// Handle type changes first, since some of the later conditions in this
|
|
|
|
// method look at mType and want to see the new value.
|
|
|
|
if (aName == nsGkAtoms::type) {
|
|
|
|
if (aValue) {
|
|
|
|
mType = aValue->GetEnumValue();
|
|
|
|
} else {
|
|
|
|
mType = kMenuItemDefaultType->value;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-08-08 21:31:32 +04:00
|
|
|
if ((aName == nsGkAtoms::radiogroup || aName == nsGkAtoms::type) &&
|
|
|
|
mType == CMD_TYPE_RADIO && !mParserCreating) {
|
2016-03-31 13:58:25 +03:00
|
|
|
if (IsInUncomposedDoc() && GetParent()) {
|
2011-08-08 21:31:32 +04:00
|
|
|
AddedToRadioGroup();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Checked must be set no matter what type of menuitem it is, since
|
|
|
|
// GetChecked() must reflect the new value
|
|
|
|
if (aName == nsGkAtoms::checked && !mCheckedDirty) {
|
|
|
|
if (mParserCreating) {
|
|
|
|
mShouldInitChecked = true;
|
|
|
|
} else {
|
|
|
|
InitChecked();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nsGenericHTMLElement::AfterSetAttr(
|
2017-10-10 00:33:38 +03:00
|
|
|
aNameSpaceID, aName, aValue, aOldValue, aSubjectPrincipal, aNotify);
|
2011-08-08 21:31:32 +04:00
|
|
|
}
|
|
|
|
|
2013-02-18 14:05:52 +04:00
|
|
|
void HTMLMenuItemElement::WalkRadioGroup(Visitor* aVisitor) {
|
2011-08-08 21:31:32 +04:00
|
|
|
nsIContent* parent = GetParent();
|
|
|
|
if (!parent) {
|
|
|
|
aVisitor->Visit(this);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-07-23 00:11:41 +03:00
|
|
|
BorrowedAttrInfo info1(GetAttrInfo(kNameSpaceID_None, nsGkAtoms::radiogroup));
|
2011-09-29 10:19:26 +04:00
|
|
|
bool info1Empty = !info1.mValue || info1.mValue->IsEmptyString();
|
2011-08-08 21:31:32 +04:00
|
|
|
|
|
|
|
for (nsIContent* cur = parent->GetFirstChild(); cur;
|
|
|
|
cur = cur->GetNextSibling()) {
|
2018-03-22 00:39:04 +03:00
|
|
|
HTMLMenuItemElement* menuitem = HTMLMenuItemElement::FromNode(cur);
|
2011-08-08 21:31:32 +04:00
|
|
|
|
|
|
|
if (!menuitem || menuitem->GetType() != CMD_TYPE_RADIO) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2016-07-23 00:11:41 +03:00
|
|
|
BorrowedAttrInfo info2(
|
|
|
|
menuitem->GetAttrInfo(kNameSpaceID_None, nsGkAtoms::radiogroup));
|
2011-09-29 10:19:26 +04:00
|
|
|
bool info2Empty = !info2.mValue || info2.mValue->IsEmptyString();
|
2011-08-08 21:31:32 +04:00
|
|
|
|
2011-12-18 10:00:47 +04:00
|
|
|
if (info1Empty != info2Empty || (info1.mValue && info2.mValue &&
|
|
|
|
!info1.mValue->Equals(*info2.mValue))) {
|
2011-08-08 21:31:32 +04:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!aVisitor->Visit(menuitem)) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-02-18 14:05:52 +04:00
|
|
|
HTMLMenuItemElement* HTMLMenuItemElement::GetSelectedRadio() {
|
|
|
|
HTMLMenuItemElement* result = nullptr;
|
2011-08-08 21:31:32 +04:00
|
|
|
|
|
|
|
GetCheckedVisitor visitor(&result);
|
|
|
|
WalkRadioGroup(&visitor);
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2013-02-18 14:05:52 +04:00
|
|
|
void HTMLMenuItemElement::AddedToRadioGroup() {
|
2011-09-29 10:19:26 +04:00
|
|
|
bool checkedDirty = mCheckedDirty;
|
2011-08-08 21:31:32 +04:00
|
|
|
if (mChecked) {
|
|
|
|
ClearCheckedVisitor visitor1(this);
|
|
|
|
GetCheckedDirtyVisitor visitor2(&checkedDirty, this);
|
|
|
|
CombinedVisitor visitor(&visitor1, &visitor2);
|
|
|
|
WalkRadioGroup(&visitor);
|
|
|
|
} else {
|
|
|
|
GetCheckedDirtyVisitor visitor(&checkedDirty, this);
|
|
|
|
WalkRadioGroup(&visitor);
|
|
|
|
}
|
|
|
|
mCheckedDirty = checkedDirty;
|
|
|
|
}
|
|
|
|
|
2013-02-18 14:05:52 +04:00
|
|
|
void HTMLMenuItemElement::InitChecked() {
|
2017-10-13 01:10:50 +03:00
|
|
|
bool defaultChecked = DefaultChecked();
|
2011-08-08 21:31:32 +04:00
|
|
|
mChecked = defaultChecked;
|
|
|
|
if (mType == CMD_TYPE_RADIO) {
|
|
|
|
ClearCheckedVisitor visitor(this);
|
|
|
|
WalkRadioGroup(&visitor);
|
|
|
|
}
|
|
|
|
}
|
2013-02-18 14:05:52 +04:00
|
|
|
|
Bug 1117172 part 3. Change the wrappercached WrapObject methods to allow passing in aGivenProto. r=peterv
The only manual changes here are to BindingUtils.h, BindingUtils.cpp,
Codegen.py, Element.cpp, IDBFileRequest.cpp, IDBObjectStore.cpp,
dom/workers/Navigator.cpp, WorkerPrivate.cpp, DeviceStorageRequestChild.cpp,
Notification.cpp, nsGlobalWindow.cpp, MessagePort.cpp, nsJSEnvironment.cpp,
Sandbox.cpp, XPCConvert.cpp, ExportHelpers.cpp, and DataStoreService.cpp. The
rest of this diff was generated by running the following commands:
find . -name "*.h" -o -name "*.cpp" | xargs perl -pi -e 'BEGIN { $/ = undef } s/(WrapObjectInternal\(JSContext *\* *(?:aCx|cx|aContext|aCtx|js))\)/\1, JS::Handle<JSObject*> aGivenProto)/g'
find . -name "*.h" -o -name "*.cpp" | xargs perl -pi -e 'BEGIN { $/ = undef } s/(WrapObjectInternal\((?:aCx|cx|aContext|aCtx|js))\)/\1, aGivenProto)/g'
find . -name "*.h" -o -name "*.cpp" | xargs perl -pi -e 'BEGIN { $/ = undef } s/(WrapNode\(JSContext *\* *(?:aCx|cx|aContext|aCtx|js))\)/\1, JS::Handle<JSObject*> aGivenProto)/g'
find . -name "*.h" -o -name "*.cpp" | xargs perl -pi -e 'BEGIN { $/ = undef } s/(WrapNode\((?:aCx|cx|aContext|aCtx|js))\)/\1, aGivenProto)/g'
find . -name "*.h" -o -name "*.cpp" | xargs perl -pi -e 'BEGIN { $/ = undef } s/(WrapObject\(JSContext *\* *(?:aCx|cx|aContext|aCtx|js))\)/\1, JS::Handle<JSObject*> aGivenProto)/g'
find . -name "*.h" -o -name "*.cpp" | xargs perl -pi -e 'BEGIN { $/ = undef } s/(Binding(?:_workers)?::Wrap\((?:aCx|cx|aContext|aCtx|js), [^,)]+)\)/\1, aGivenProto)/g'
2015-03-19 17:13:33 +03:00
|
|
|
JSObject* HTMLMenuItemElement::WrapNode(JSContext* aCx,
|
|
|
|
JS::Handle<JSObject*> aGivenProto) {
|
2018-06-26 00:20:54 +03:00
|
|
|
return HTMLMenuItemElement_Binding::Wrap(aCx, this, aGivenProto);
|
2013-02-18 14:06:27 +04:00
|
|
|
}
|
|
|
|
|
2013-02-18 14:05:52 +04:00
|
|
|
} // namespace dom
|
|
|
|
} // namespace mozilla
|
2013-11-19 23:21:29 +04:00
|
|
|
|
|
|
|
#undef NS_ORIGINAL_CHECKED_VALUE
|