mirror of
https://github.com/roytam1/palemoon27.git
synced 2026-05-26 14:18:48 +00:00
d64e96b4a0
- Bug 448064 - Add crashtest. (d7e9a3f9bb)
- Minor cleanup (cbab3e1ede)
- Bug 1222829: Remove URIChecker. r=mcmanus (74c16ae58c)
- Bug 1220682 - Clear exceptions on single-arg init. r=bz # Please enter the commit message for your changes. Lines starting (4736e0fb57)
- Bug 1229237 (part 1) - Make nsIWidget::{Create,CreateChildren}() take a LayoutDeviceIntRect. r=botond. (992ec123ff)
- Bug 1229237 (part 2) - Make nsIWidget::DrawWindowUnderlay() take a LayoutDeviceIntRect. r=botond. (ad4f13440d)
- Bug 1229237 (part 3) - Make nsIWidget::Invalidate() take a LayoutDeviceIntRect. r=botond. (19f10846a3)
- Bug 1229237 (part 4) - Make ThemeGeometry::mRect a LayoutDeviceIntRect. r=botond. (5f17f6be14)
- Bug 1228125 (part 1) - Remove nsIWidget::GetBoundsUntyped(). r=botond. (5e4d0811e7)
- Bug 1228125 (part 2) - Remove nsIWidget::GetScreenBoundsUntyped(). r=botond. (3ac4b3271b)
- Bug 1228125 (part 3) - Remove nsIWidget::GetClientBoundsUntyped(). r=botond. (7163a1a6e6)
- Bug 1229237 (part 5) - Make Update{Opaque,WindowDragging}Region() take a LayoutDeviceIntRegion. r=botond. (907aafd4c0)
- missing bit Bug 1229237 (part 1) (c160f227b3)
- missing bit Bug 1229237 (part 1) (edc4237028)
- Bug 1215959 - (GeckoCaret2) update experimental prefs, r=snorp (df3a19e142)
- Bug 1168881 - Enabling AccessibleCarets crashes Firefox when trying to input text, r=tylin (abbd401672)
- Bug 1215959 - (GeckoCaret2) Remove experimental bits added to Touch/Selection carets, r=tylin (291385ad0e)
- Bug 1219236 - Part 1: Flush throttle styles for all descendant sub documents and *root* document itself. r=roc (b6c4d4c815)
- Bug 1153130 - Prevent FireFox crashes when pointer events are enabled. r=smaug (a7093fae15)
- Bug 1226041 - Remove mFirstPaint hack for JPZC when the C++ APZ is enabled. r=kats (6af6102f1f)
- Bug 1223228 - Remove unused nsFrameManagerBase::mStyleSet. r=dholbert (9145f9cb38)
- Bug 1189353 - Change NS_ macro prefix to MOZ_ in AccessibleCaretEventHub. r=mtseng (820920546e)
- Bug 1210315 - Use preference to control whether to use long tap injector. r=roc (00ee7e8874)
- Bug 1206545 - Store nsIPresShell pointer in constructor instead of Init(). r=roc (6a9f8216f7)
- Bug 1194063 - Update link to point to the diagram directly. r=mtseng (3d627f37da)
- Bug 1225701 - Update comments in AccessibleCaret files. r=mtseng (fe1d296655)
- Bug 1211365 - Remove NS_IMPL_STATE_UTILITIES. r=mtseng (79713764b9)
- Bug 1211365 - Make deleted functions in class State public. r=mtseng (da2df347b3)
- Bug 1211365 - Delete wheel event handling. r=mtseng (7bacb7e8f5)
- Bug 1200194 - Tweak formatting and debug logging. r=roc (148a22ce3c)
- Bug 1082425 - Commit composition string before changing focus by long tap. r=masayuki (1452e46511)
- Bug 1196176 - No need to test mLastUpdateCaretMode. r=mtseng (fc93c83698)
- Bug 1196176 - Hide carets for mouse down reason. r=mtseng (3c13762b86)
- Bug 1209841 - Merge test_selectioncarets2.py into test_selectioncarets.py. r=automatedtester (35575914b9)
- Bug 1209841 - Simplify functions related to select word. r=automatedtester (56bf6052e1)
- Bug 1209841 - Send synthesized mouse long tap to gecko. r=automatedtester (e83509d49b)
- Bug 1210315 - Inline selectors in carets tests. r=automatedtester (cebad9b688)
- Bug 1163490 - Remove SpecialPowers from marionette carets test. r=jgriffin (fd1e4b6442)
- Bug 1210315 - Use pref utilities in test_selectioncarets.py. r=automatedtester (09efee2a80)
- Bug 1207934 - Add marionette test for appearance changing. r=automatedtester (6e1c5aea60)
- Bug 1210315 - Use pref utilities in test_touchcaret.py. r=automatedtester (a17a92808e)
- Bug 1200364 - Fix AccessibleCaret jumps when dragging. r=roc (66c36347f7)
- Bug 1203074: Move caret to end of textual input field before sending keys (1a8cf55bd7)
- Bug 1164233 - Enlarge the timeout margin from 1.5x to 3x. r=mtseng (188b17550d)
- Bug 1196176 - Fix CaretStateChanged not dispatch on empty content. r=mtseng (5d6930c1e1)
- Bug 1196176 - Do not fire extra CaretStateChanged event when typing. r=mtseng (93f4ccdc97)
- Bug 1211365 - Add test case for scrolling in selection mode. r=mtseng (bfd69f4337)
- Bug 1212732 - Update caret in cursor mode on scroll-end if it was logically visible. r=roc (07ddba2c15)
- Bug 1215959 - (GeckoCaret2) Upgrade Core and AccessibleCaret, r=smaug (fe388a4b37)
- Bug 1216857 - Notify selection dialog to update its position if carets are logically visible. r=TYLin (e759cbe5cd)
- Bug 1215798 nsContentIterator::Init(nsIDOMRange*) should not skip empty start node when mPre is true r=smaug (013857d99d)
- Bug 1215816 nsContentIterator::Init(nsIDOMRange*) should not include end node if it's an empty element and the end offset is 0 when mPre is true r=smaug (5276c84afb)
372 lines
11 KiB
C++
372 lines
11 KiB
C++
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
/* 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/. */
|
|
|
|
#include "nsCOMPtr.h"
|
|
#include "nsIRootBox.h"
|
|
#include "nsIPresShell.h"
|
|
#include "nsIContent.h"
|
|
#include "nsNameSpaceManager.h"
|
|
#include "nsGkAtoms.h"
|
|
#include "nsMenuPopupFrame.h"
|
|
#include "nsView.h"
|
|
#include "mozilla/AppUnits.h"
|
|
#include "mozilla/dom/DOMRect.h"
|
|
#include "mozilla/dom/PopupBoxObject.h"
|
|
#include "mozilla/dom/Element.h"
|
|
#include "mozilla/dom/Event.h"
|
|
#include "mozilla/dom/PopupBoxObjectBinding.h"
|
|
|
|
namespace mozilla {
|
|
namespace dom {
|
|
|
|
NS_IMPL_ADDREF_INHERITED(PopupBoxObject, BoxObject)
|
|
NS_IMPL_RELEASE_INHERITED(PopupBoxObject, BoxObject)
|
|
NS_INTERFACE_MAP_BEGIN(PopupBoxObject)
|
|
NS_INTERFACE_MAP_END_INHERITING(BoxObject)
|
|
|
|
PopupBoxObject::PopupBoxObject()
|
|
{
|
|
}
|
|
|
|
PopupBoxObject::~PopupBoxObject()
|
|
{
|
|
}
|
|
|
|
nsIContent* PopupBoxObject::GetParentObject() const
|
|
{
|
|
return BoxObject::GetParentObject();
|
|
}
|
|
|
|
JSObject* PopupBoxObject::WrapObject(JSContext* aCx, JS::Handle<JSObject*> aGivenProto)
|
|
{
|
|
return PopupBoxObjectBinding::Wrap(aCx, this, aGivenProto);
|
|
}
|
|
|
|
nsPopupSetFrame*
|
|
PopupBoxObject::GetPopupSetFrame()
|
|
{
|
|
nsIRootBox* rootBox = nsIRootBox::GetRootBox(GetPresShell(false));
|
|
if (!rootBox)
|
|
return nullptr;
|
|
|
|
return rootBox->GetPopupSetFrame();
|
|
}
|
|
|
|
void
|
|
PopupBoxObject::HidePopup(bool aCancel)
|
|
{
|
|
nsXULPopupManager* pm = nsXULPopupManager::GetInstance();
|
|
if (pm && mContent) {
|
|
pm->HidePopup(mContent, false, true, false, aCancel);
|
|
}
|
|
}
|
|
|
|
void
|
|
PopupBoxObject::ShowPopup(Element* aAnchorElement,
|
|
Element& aPopupElement,
|
|
int32_t aXPos, int32_t aYPos,
|
|
const nsAString& aPopupType,
|
|
const nsAString& aAnchorAlignment,
|
|
const nsAString& aPopupAlignment)
|
|
{
|
|
nsXULPopupManager* pm = nsXULPopupManager::GetInstance();
|
|
if (pm && mContent) {
|
|
nsCOMPtr<nsIContent> anchorContent(do_QueryInterface(aAnchorElement));
|
|
nsAutoString popupType(aPopupType);
|
|
nsAutoString anchor(aAnchorAlignment);
|
|
nsAutoString align(aPopupAlignment);
|
|
pm->ShowPopupWithAnchorAlign(mContent, anchorContent, anchor, align,
|
|
aXPos, aYPos,
|
|
popupType.EqualsLiteral("context"));
|
|
}
|
|
}
|
|
|
|
void
|
|
PopupBoxObject::OpenPopup(Element* aAnchorElement,
|
|
const nsAString& aPosition,
|
|
int32_t aXPos, int32_t aYPos,
|
|
bool aIsContextMenu,
|
|
bool aAttributesOverride,
|
|
Event* aTriggerEvent)
|
|
{
|
|
nsXULPopupManager* pm = nsXULPopupManager::GetInstance();
|
|
if (pm && mContent) {
|
|
nsCOMPtr<nsIContent> anchorContent(do_QueryInterface(aAnchorElement));
|
|
pm->ShowPopup(mContent, anchorContent, aPosition, aXPos, aYPos,
|
|
aIsContextMenu, aAttributesOverride, false, aTriggerEvent);
|
|
}
|
|
}
|
|
|
|
void
|
|
PopupBoxObject::OpenPopupAtScreen(int32_t aXPos, int32_t aYPos,
|
|
bool aIsContextMenu,
|
|
Event* aTriggerEvent)
|
|
{
|
|
nsXULPopupManager* pm = nsXULPopupManager::GetInstance();
|
|
if (pm && mContent)
|
|
pm->ShowPopupAtScreen(mContent, aXPos, aYPos, aIsContextMenu, aTriggerEvent);
|
|
}
|
|
|
|
void
|
|
PopupBoxObject::OpenPopupAtScreenRect(const nsAString& aPosition,
|
|
int32_t aXPos, int32_t aYPos,
|
|
int32_t aWidth, int32_t aHeight,
|
|
bool aIsContextMenu,
|
|
bool aAttributesOverride,
|
|
Event* aTriggerEvent)
|
|
{
|
|
nsXULPopupManager* pm = nsXULPopupManager::GetInstance();
|
|
if (pm && mContent) {
|
|
pm->ShowPopupAtScreenRect(mContent, aPosition,
|
|
nsIntRect(aXPos, aYPos, aWidth, aHeight),
|
|
aIsContextMenu, aAttributesOverride, aTriggerEvent);
|
|
}
|
|
}
|
|
|
|
void
|
|
PopupBoxObject::MoveTo(int32_t aLeft, int32_t aTop)
|
|
{
|
|
nsMenuPopupFrame *menuPopupFrame = mContent ? do_QueryFrame(mContent->GetPrimaryFrame()) : nullptr;
|
|
if (menuPopupFrame) {
|
|
menuPopupFrame->MoveTo(CSSIntPoint(aLeft, aTop), true);
|
|
}
|
|
}
|
|
|
|
void
|
|
PopupBoxObject::MoveToAnchor(Element* aAnchorElement,
|
|
const nsAString& aPosition,
|
|
int32_t aXPos, int32_t aYPos,
|
|
bool aAttributesOverride)
|
|
{
|
|
if (mContent) {
|
|
nsCOMPtr<nsIContent> anchorContent(do_QueryInterface(aAnchorElement));
|
|
|
|
nsMenuPopupFrame *menuPopupFrame = do_QueryFrame(mContent->GetPrimaryFrame());
|
|
if (menuPopupFrame && menuPopupFrame->IsVisible()) {
|
|
menuPopupFrame->MoveToAnchor(anchorContent, aPosition, aXPos, aYPos, aAttributesOverride);
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
PopupBoxObject::SizeTo(int32_t aWidth, int32_t aHeight)
|
|
{
|
|
if (!mContent)
|
|
return;
|
|
|
|
nsAutoString width, height;
|
|
width.AppendInt(aWidth);
|
|
height.AppendInt(aHeight);
|
|
|
|
nsCOMPtr<nsIContent> content = mContent;
|
|
|
|
// We only want to pass aNotify=true to SetAttr once, but must make sure
|
|
// we pass it when a value is being changed. Thus, we check if the height
|
|
// is the same and if so, pass true when setting the width.
|
|
bool heightSame = content->AttrValueIs(kNameSpaceID_None, nsGkAtoms::height, height, eCaseMatters);
|
|
|
|
content->SetAttr(kNameSpaceID_None, nsGkAtoms::width, width, heightSame);
|
|
content->SetAttr(kNameSpaceID_None, nsGkAtoms::height, height, true);
|
|
}
|
|
|
|
bool
|
|
PopupBoxObject::AutoPosition()
|
|
{
|
|
nsMenuPopupFrame *menuPopupFrame = mContent ? do_QueryFrame(mContent->GetPrimaryFrame()) : nullptr;
|
|
if (menuPopupFrame) {
|
|
return menuPopupFrame->GetAutoPosition();
|
|
}
|
|
return true;
|
|
}
|
|
|
|
void
|
|
PopupBoxObject::SetAutoPosition(bool aShouldAutoPosition)
|
|
{
|
|
nsMenuPopupFrame *menuPopupFrame = mContent ? do_QueryFrame(mContent->GetPrimaryFrame()) : nullptr;
|
|
if (menuPopupFrame) {
|
|
menuPopupFrame->SetAutoPosition(aShouldAutoPosition);
|
|
}
|
|
}
|
|
|
|
void
|
|
PopupBoxObject::EnableRollup(bool aShouldRollup)
|
|
{
|
|
// this does nothing now
|
|
}
|
|
|
|
void
|
|
PopupBoxObject::SetConsumeRollupEvent(uint32_t aConsume)
|
|
{
|
|
nsMenuPopupFrame *menuPopupFrame = do_QueryFrame(GetFrame(false));
|
|
if (menuPopupFrame) {
|
|
menuPopupFrame->SetConsumeRollupEvent(aConsume);
|
|
}
|
|
}
|
|
|
|
void
|
|
PopupBoxObject::EnableKeyboardNavigator(bool aEnableKeyboardNavigator)
|
|
{
|
|
if (!mContent)
|
|
return;
|
|
|
|
// Use ignorekeys="true" on the popup instead of using this function.
|
|
if (aEnableKeyboardNavigator)
|
|
mContent->UnsetAttr(kNameSpaceID_None, nsGkAtoms::ignorekeys, true);
|
|
else
|
|
mContent->SetAttr(kNameSpaceID_None, nsGkAtoms::ignorekeys,
|
|
NS_LITERAL_STRING("true"), true);
|
|
}
|
|
|
|
void
|
|
PopupBoxObject::GetPopupState(nsString& aState)
|
|
{
|
|
// set this here in case there's no frame for the popup
|
|
aState.AssignLiteral("closed");
|
|
|
|
nsMenuPopupFrame *menuPopupFrame = mContent ? do_QueryFrame(mContent->GetPrimaryFrame()) : nullptr;
|
|
if (menuPopupFrame) {
|
|
switch (menuPopupFrame->PopupState()) {
|
|
case ePopupShown:
|
|
aState.AssignLiteral("open");
|
|
break;
|
|
case ePopupShowing:
|
|
case ePopupOpening:
|
|
case ePopupVisible:
|
|
aState.AssignLiteral("showing");
|
|
break;
|
|
case ePopupHiding:
|
|
case ePopupInvisible:
|
|
aState.AssignLiteral("hiding");
|
|
break;
|
|
case ePopupClosed:
|
|
break;
|
|
default:
|
|
NS_NOTREACHED("Bad popup state");
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
nsINode*
|
|
PopupBoxObject::GetTriggerNode() const
|
|
{
|
|
nsMenuPopupFrame *menuPopupFrame = mContent ? do_QueryFrame(mContent->GetPrimaryFrame()) : nullptr;
|
|
return nsMenuPopupFrame::GetTriggerContent(menuPopupFrame);
|
|
}
|
|
|
|
Element*
|
|
PopupBoxObject::GetAnchorNode() const
|
|
{
|
|
nsMenuPopupFrame *menuPopupFrame = mContent ? do_QueryFrame(mContent->GetPrimaryFrame()) : nullptr;
|
|
if (!menuPopupFrame) {
|
|
return nullptr;
|
|
}
|
|
|
|
nsIContent* anchor = menuPopupFrame->GetAnchor();
|
|
return anchor && anchor->IsElement() ? anchor->AsElement() : nullptr;
|
|
}
|
|
|
|
already_AddRefed<DOMRect>
|
|
PopupBoxObject::GetOuterScreenRect()
|
|
{
|
|
RefPtr<DOMRect> rect = new DOMRect(mContent);
|
|
|
|
// Return an empty rectangle if the popup is not open.
|
|
nsMenuPopupFrame *menuPopupFrame = do_QueryFrame(GetFrame(false));
|
|
if (!menuPopupFrame || !menuPopupFrame->IsOpen()) {
|
|
return rect.forget();
|
|
}
|
|
|
|
nsView* view = menuPopupFrame->GetView();
|
|
if (view) {
|
|
nsIWidget* widget = view->GetWidget();
|
|
if (widget) {
|
|
LayoutDeviceIntRect screenRect;
|
|
widget->GetScreenBounds(screenRect);
|
|
|
|
int32_t pp = menuPopupFrame->PresContext()->AppUnitsPerDevPixel();
|
|
rect->SetLayoutRect(LayoutDeviceIntRect::ToAppUnits(screenRect, pp));
|
|
}
|
|
}
|
|
return rect.forget();
|
|
}
|
|
|
|
void
|
|
PopupBoxObject::GetAlignmentPosition(nsString& positionStr)
|
|
{
|
|
positionStr.Truncate();
|
|
|
|
// This needs to flush layout.
|
|
nsMenuPopupFrame *menuPopupFrame = do_QueryFrame(GetFrame(true));
|
|
if (!menuPopupFrame)
|
|
return;
|
|
|
|
int8_t position = menuPopupFrame->GetAlignmentPosition();
|
|
switch (position) {
|
|
case POPUPPOSITION_AFTERSTART:
|
|
positionStr.AssignLiteral("after_start");
|
|
break;
|
|
case POPUPPOSITION_AFTEREND:
|
|
positionStr.AssignLiteral("after_end");
|
|
break;
|
|
case POPUPPOSITION_BEFORESTART:
|
|
positionStr.AssignLiteral("before_start");
|
|
break;
|
|
case POPUPPOSITION_BEFOREEND:
|
|
positionStr.AssignLiteral("before_end");
|
|
break;
|
|
case POPUPPOSITION_STARTBEFORE:
|
|
positionStr.AssignLiteral("start_before");
|
|
break;
|
|
case POPUPPOSITION_ENDBEFORE:
|
|
positionStr.AssignLiteral("end_before");
|
|
break;
|
|
case POPUPPOSITION_STARTAFTER:
|
|
positionStr.AssignLiteral("start_after");
|
|
break;
|
|
case POPUPPOSITION_ENDAFTER:
|
|
positionStr.AssignLiteral("end_after");
|
|
break;
|
|
case POPUPPOSITION_OVERLAP:
|
|
positionStr.AssignLiteral("overlap");
|
|
break;
|
|
case POPUPPOSITION_AFTERPOINTER:
|
|
positionStr.AssignLiteral("after_pointer");
|
|
break;
|
|
default:
|
|
// Leave as an empty string.
|
|
break;
|
|
}
|
|
}
|
|
|
|
int32_t
|
|
PopupBoxObject::AlignmentOffset()
|
|
{
|
|
nsMenuPopupFrame *menuPopupFrame = do_QueryFrame(GetFrame(false));
|
|
if (!menuPopupFrame)
|
|
return 0;
|
|
|
|
int32_t pp = mozilla::AppUnitsPerCSSPixel();
|
|
// Note that the offset might be along either the X or Y axis, but for the
|
|
// sake of simplicity we use a point with only the X axis set so we can
|
|
// use ToNearestPixels().
|
|
nsPoint appOffset(menuPopupFrame->GetAlignmentOffset(), 0);
|
|
nsIntPoint popupOffset = appOffset.ToNearestPixels(pp);
|
|
return popupOffset.x;
|
|
}
|
|
|
|
} // namespace dom
|
|
} // namespace mozilla
|
|
|
|
// Creation Routine ///////////////////////////////////////////////////////////////////////
|
|
|
|
nsresult
|
|
NS_NewPopupBoxObject(nsIBoxObject** aResult)
|
|
{
|
|
*aResult = new mozilla::dom::PopupBoxObject();
|
|
NS_ADDREF(*aResult);
|
|
return NS_OK;
|
|
}
|