mirror of
https://github.com/roytam1/palemoon27.git
synced 2026-05-26 14:18:48 +00:00
8a0a002cf2
- Bug 1148708: Add missing 'override' annotations in DocAccessibleChild.h. rs=ehsan (d606358545)
- Bug 1210408 - make nsMaiInterfaceAction work with proxies, r=tbsaunde (f7c819c6ae)
- Bug 1210407 - teach nsMaiInterfaceTable to use proxies, r=tbsaunde (4ca4f10b5f)
- bug 1185157 make sure we don't send an event to a destroyed ipc document r=billm (23acf53f75)
- bug 1214864 - make SetCarretOffset() async r=davidb (e3079e9b2d)
- missing of Bug 1139972 - IPC Proxy for charAt, r=tbsaunde (e9593ed752)
- bug 1191598 - Pass MOZ_CURRENT_PROJECT in environment when running post-build automation steps for universal mac builds. r=gps (fc342c6ced)
- Bug 1164596 - Add mach android-emulator command; r=ahal (afeb9b27d1)
- Bug 1223149 - Add basic usage documentation for mach build; r=glandium (bfb802d175)
- Bug 1182301 - Improve 'mach build' notifications. r=gps (2c65a122d1)
- Bug 1184696 - Add clobber targets to |mach clobber|; Ability to clobber compiled python files, r=gps (35d8be292e)
- Bug 1117958 - Allow any debugging options to the run or gtest mach subcommands to automatically enable debugging. r=gps (32f986af4b)
- Bug 1180081 - Properly rebuild gtest/libxul before running gtests. r=gps (80db9a3d49)
- Bug 1171647, part 1 - Define a new function to convert the mode to a string. r=njn (61ad16f5ba)
- Bug 1171647, part 2 - Remove redundant assertion for dark matter mode. r=njn (b5ac9519f3)
- Bug 1058178, part 1 - Implement DMD heap scanning mode. r=njn (60e1079536)
- Bug 1058178, part 2 - Implement address clamping analysis for DMD scan logs. r=njn (45c0326b93)
- Bug 1102388 - Fix DMD static constructor ordering dependency. r=mccr8 (59b87897a1)
- Bug 1128705 - Don't redefine PAGE_SIZE in DMD if it's already defined. r=erahm (49216348ee)
- Bug 1179042 - Add a script for analyzing memory blocks using a heap scan DMD log. r=njn DONTBUILD (1c08d2d66e)
- Bug 1207245 - part 6 - rename nsRefPtr<T> to RefPtr<T>; r=ehsan; a=Tomcat (1c999d139b)
- Bug 1158772 - fix non-idiomatic memset call in nsDeque.cpp; r=erahm (de6b555245)
- Bug 1199400 - Part 1: Use CheckedInt when growing nsDeque capacity. r=froydnj (dfdf6814a3)
- Bug 1199400 - Part 0: Remove unused nsDequeIterator. r=froydnj (38d69d7f47)
- Bug 1199400 - Part 2: Add tests for possible nsDeque corner cases. r=froydnj (931baff195)
- Bug 1201997 - Part 1 - Converted compiled test to gtest for nsDeque class. Added tests to test untested methods. r=froydn (e893916651)
- Bug 1201997 - Part 2 - Removing unused methods from the nsDeque class. r=froydn (41595a90ac)
- Bug 1201997 - Part 3 - Make internally used methods private. r=froydn (8cd3afd96f)
- Bug 1201997 - Part 4 - Change size and offset variables to size_t.r=froydn (73eabc8d60)
- Bug 1215140 P1 Add an nsIConsoleReportCollector interface to support navigation channel logging. r=bz (8a41535e2b)
- Bug 1215140 P2 Make HttpBaseChannel implement nsIConsoleReportCollector. r=bz (75fca301f2)
- Bug 1197679 - If nsUnknownDecoder is involved in e10s DivertToParent can break. r=jduell (5d94a12504)
- Bug 1178991 - smartptr for http converter r=hurley (8e7fbc8443)
- bug 366559 - patch 2, fix nsHTTPCompressConv indentation r=bagder (ba762da587)
- bug 366559 - patch 3, fix nsHTTPCompressConv bracing style r=bagder (54195ab451)
- bits of bug 366559 - patch 7, content-encoding brotli for http (f0b4051022)
- Bug 1205112 - Make PushEvent.data nullable. r=mt,smaug (775db32856)
- Bug 1193414 - SharedWorkers thread should be kept alive also when the SharedWorker object is CCed, r=khuey (b77ea8125c)
- Bug 1206520: Add about:config prefs to enable throwing on asm.js validation failures; r=bz (c42126665d)
- Bug 1193414 - Telemetry for SharedWorker spawning. r=bkelly (77984b7bcc)
- Bug 1205676 - Enable WPT service-worker/unregister-then-register-new-script.https.html in e10s, r=nsm (ec24939cf6)
- Bug 1193133 - Throw when calling postMessage from a Service Worker dom object with no global. r=bkelly (526dcacfab)
- Bug 1181871 P1 Only enforce Cache Context shared data destruction on target thread after init. r=ehsan (cdbf3ed3a8)
- Bug 1181871 P2 Fix ServiceWorkerManager usage of stack-based ErrorResult. r=ehsan (c449195d90)
- minor cleanup and missing bit of 1198230 (02f459db05)
- Bug 1143717 - Implement the ServiceWorkerMessageEvent interface. r=baku (027b3465f2)
- fix misspatch (708eee4e84)
- Bug 1188545 - Disentangle service workers from shared workers and refactor event dispatching code into a separate class. r=nsm,mrbkap (fb5b5341c9)
- Bug 1205228 - Change PackagedAppVerifier to notify the verification result asynchronously. r=valentin. (9edda0fa00)
- Bug 1178518 - Packaged App Utils. r=valentin (f60f3b7a93)
- Bug 1213150 - Part 1: Add a nsContentUtils::IsNonSubresourceRequest helper; r=jdm (b509cc3cc9)
- Bug 1213150 - Part 2: Rework ShouldPrepareForIntercept() in terms of subresource requests; r=jdm (2e92fe8780)
- Bug 1213150 - Part 3: Remove nsIInterceptedChannel.isNavigation; r=jdm (becf1cc12f)
- Bug 1213150 follow-up: fix build bustage (8d73d6ca73)
- Bug 1198394 - Part 1: Allow interception of HSTS upgraded connections in non-e10s mode; r=mcmanus (f504c5be08)
- Bug 1198394 - Part 2: Add a test for interception of HSTS upgraded connections; r=jdm (054e984eef)
- Bug 1187011 - Don't allow response body with null body status. r=bkelly (b1860741d1)
- missing bit of 1140788 (29d319712e)
- Bug 1213436 - Reject core dumps with node IDs that don't fit in an IEEE 754 double; r=sfink (3c1f6fdda0)
- Bug 1211006 - Add Debugger.Source.prototype.canonicalId; r=ejpbruel (eef7b79fce)
- Bug 1199218 - Implement JS::ubi::Node::size for js::LazyScript referents; r=sfink (098a48d240)
- Bug 1220031 - Add JS::ubi::Node::scriptFilename; r=sfink (6b824ae680)
- Bug 1143575. Remove unused MediaQueue::Empty. r=cpearce (de737f3433)
- Bug 1209933 - Make sure all parent runtime pointers are the topmost parent, r=billm. (fe824d967d)
- Bug 1197012 - Fix ThrowTypeError in Notification. r=mccr8 (0b1a097526)
- Bug 1197893 - Check the number of arguments for ThrowTypeError() and ThrowRangeError() at compile time. r=peterv (d98c7d78a0)
- Bug 1142083 - Add test for IDN Unicode domain redirect. r=mcmanus (0c8961fe17)
- Bug 1187159 - Add mochitest for loading packaged apps (iframe+fetch+mozapp) r=jduell (ce90ea561b)
- Bug 1186290 - Notify TabParent to switch process when loading a signed package. r=honzab, r=kanru. (c58a14554a)
- fix (15e2df75eb)
- Bug 1206124 P1 Fix "same-origin" CORS credentials in FetchDriver. r=ehsan (fae1bb6ab3)
- Bug 1206124 P2 Test fetch() with credentials and redirects. r=ehsan (ffc6254112)
- Bug 1211751: Remove nsIChannelEventSink-forwarding from EventSource and FetchDriver. It's never needed. r=smaug (adafe5737a)
- Bug 1212433 Fail fetch() calls that require preflight and also redirect. r=sicking a=abillings (c0d6742b9e)
- Bug 1193128 - Fix base64 decoding when fetching data URIs. r=baku (80bafa291a)
- Bug 1195167 part 1: Let necko handle all protocols. r=bkelly (bb932b0ada)
- Bug 1195167 part 2: Remove redundant aCORSFlag argument and instead use mCORSFlagEverSet. r=bkelly (beadafcad0)
- Bug 1195167 part 3: Remove more scheme-specific handling from FetchDriver. r=bkelly (d00b38db9e)
- Bug 1195167 part 4: Remove FetchDriver::BasicFetch since it is empty. r=bkelly (c5ed097267)
- Bug 1210413 P2 Test CORS credentials on cross-origin redirects. r=sicking a=dveditz (b4eeb8aac0)
- Bug 1210413 P1 Propagate new channel load flags from child to parent on redirect. r=jduell a=dveditz (8b329af4fa)
- Bug 1195167 part 5: Make FetchDriver use AsyncOpen2. r=bkelly (cc217c4cc1)
- Bug 1195167 part 6: Some code simplification since necko handles fetch recursion. r=bkelly (f3b6da2262)
- Bug 1195167: Followup to fix test which I forgot to change (81e7439a2e)
- Bug 1215746: Remove RequestMode::Cors_with_forced_preflight. r=bkelly (0336e812b6)
- Bug 1211000: Move CORS preflight logic from nsCORSListenerProxy to nsCORSPreflightListener. r=ehsan (bf2f71cf22)
- missing bit of Bug 1211443 - Drop scheduled update if decoder initialization isn't done yet. r=jya (f6bc074e33)
- Bug 1182571: Fix nsILoadInfo->GetContentPolicyType API to be less ambigious. Audit and fix all users of it. r=ckerschb (5af6fa7442)
- fix (e40c8e7625)
- Bug 1173811 - Part 1: Propagate the response URL to intercepted channels when necessary (non-e10s). r=mayhemer,bkelly (26f4f13c28)
- Bug 1173811 - Part 2: Propagate the response URL to intercepted channels when necessary (e10s). r=mayhemer,bkelly (a603fe1df2)
- Bug 1154309 - Add New Resource Timing Fields r=bz,hurley (1d14eb6bef)
- Bug 1175685 - add OriginAttribute to LoadInfo. r=jonas, r=ckerschb, r=michal (a5d18bb637)
- Bug 1175685 - add OriginAttribute to LoadInfo. r=jonas, r=ckerschb, r=michal (fb07d2c8aa)
- Bug 1212904 P1 Add a LoadTainting enumeration. r=jduell (a1db8a3e99)
- Bug 1212904 P2 Add LoadTainting information to nsILoadInfo. r=jduell (2482e5e334)
- Bug 1221151 - use [infallible] in nsILoadInfo.idl instead of manual %{C++ blocks; r=jduell (aae73129b6)
- Bug 1045891 - CSP 2 child-src implementation r=ckerschb (792920aeb9)
- Bug 1219931 - CSP: Don't allow removing a policy (r=sicking) (9daaab4186)
- Bug 1208661 - Dump client-side layer textures. r=BenWa (1f2d17d515)
671 lines
22 KiB
C++
671 lines
22 KiB
C++
/* 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 <math.h>
|
|
|
|
#include "mozilla/Preferences.h"
|
|
#include "mozilla/dom/Selection.h"
|
|
#include "mozilla/dom/Element.h"
|
|
#include "mozilla/mozalloc.h"
|
|
#include "nsAString.h"
|
|
#include "nsAlgorithm.h"
|
|
#include "nsAutoPtr.h"
|
|
#include "nsCOMPtr.h"
|
|
#include "nsComputedDOMStyle.h"
|
|
#include "nsDebug.h"
|
|
#include "nsEditRules.h"
|
|
#include "nsEditor.h"
|
|
#include "nsEditorUtils.h"
|
|
#include "nsError.h"
|
|
#include "nsGkAtoms.h"
|
|
#include "nsHTMLCSSUtils.h"
|
|
#include "nsHTMLEditRules.h"
|
|
#include "nsHTMLEditUtils.h"
|
|
#include "nsHTMLEditor.h"
|
|
#include "nsHTMLObjectResizer.h"
|
|
#include "nsIContent.h"
|
|
#include "nsROCSSPrimitiveValue.h"
|
|
#include "nsIDOMCSSStyleDeclaration.h"
|
|
#include "nsIDOMElement.h"
|
|
#include "nsIDOMEventListener.h"
|
|
#include "nsIDOMEventTarget.h"
|
|
#include "nsIDOMNode.h"
|
|
#include "nsDOMCSSRGBColor.h"
|
|
#include "nsIDOMWindow.h"
|
|
#include "nsIEditor.h"
|
|
#include "nsIHTMLEditor.h"
|
|
#include "nsIHTMLObjectResizer.h"
|
|
#include "nsINode.h"
|
|
#include "nsIPresShell.h"
|
|
#include "nsISupportsImpl.h"
|
|
#include "nsISupportsUtils.h"
|
|
#include "nsLiteralString.h"
|
|
#include "nsReadableUtils.h"
|
|
#include "nsString.h"
|
|
#include "nsStringFwd.h"
|
|
#include "nsTextEditRules.h"
|
|
#include "nsTextEditUtils.h"
|
|
#include "nscore.h"
|
|
#include <algorithm>
|
|
|
|
using namespace mozilla;
|
|
using namespace mozilla::dom;
|
|
|
|
#define BLACK_BG_RGB_TRIGGER 0xd0
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLEditor::AbsolutePositionSelection(bool aEnabled)
|
|
{
|
|
nsAutoEditBatch beginBatching(this);
|
|
nsAutoRules beginRulesSniffing(this,
|
|
aEnabled ? EditAction::setAbsolutePosition :
|
|
EditAction::removeAbsolutePosition,
|
|
nsIEditor::eNext);
|
|
|
|
// the line below does not match the code; should it be removed?
|
|
// Find out if the selection is collapsed:
|
|
RefPtr<Selection> selection = GetSelection();
|
|
NS_ENSURE_TRUE(selection, NS_ERROR_NULL_POINTER);
|
|
|
|
nsTextRulesInfo ruleInfo(aEnabled ? EditAction::setAbsolutePosition :
|
|
EditAction::removeAbsolutePosition);
|
|
bool cancel, handled;
|
|
// Protect the edit rules object from dying
|
|
nsCOMPtr<nsIEditRules> kungFuDeathGrip(mRules);
|
|
nsresult res = mRules->WillDoAction(selection, &ruleInfo, &cancel, &handled);
|
|
if (NS_FAILED(res) || cancel)
|
|
return res;
|
|
|
|
return mRules->DidDoAction(selection, &ruleInfo, res);
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLEditor::GetAbsolutelyPositionedSelectionContainer(nsIDOMElement **_retval)
|
|
{
|
|
nsCOMPtr<nsIDOMElement> element;
|
|
nsresult res = GetSelectionContainer(getter_AddRefs(element));
|
|
NS_ENSURE_SUCCESS(res, res);
|
|
|
|
nsAutoString positionStr;
|
|
nsCOMPtr<nsINode> node = do_QueryInterface(element);
|
|
nsCOMPtr<nsIDOMNode> resultNode;
|
|
|
|
while (!resultNode && node && !node->IsHTMLElement(nsGkAtoms::html)) {
|
|
res = mHTMLCSSUtils->GetComputedProperty(*node, *nsGkAtoms::position,
|
|
positionStr);
|
|
NS_ENSURE_SUCCESS(res, res);
|
|
if (positionStr.EqualsLiteral("absolute"))
|
|
resultNode = GetAsDOMNode(node);
|
|
else {
|
|
node = node->GetParentNode();
|
|
}
|
|
}
|
|
|
|
element = do_QueryInterface(resultNode );
|
|
*_retval = element;
|
|
NS_IF_ADDREF(*_retval);
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLEditor::GetSelectionContainerAbsolutelyPositioned(bool *aIsSelectionContainerAbsolutelyPositioned)
|
|
{
|
|
*aIsSelectionContainerAbsolutelyPositioned = (mAbsolutelyPositionedObject != nullptr);
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLEditor::GetAbsolutePositioningEnabled(bool * aIsEnabled)
|
|
{
|
|
*aIsEnabled = mIsAbsolutelyPositioningEnabled;
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLEditor::SetAbsolutePositioningEnabled(bool aIsEnabled)
|
|
{
|
|
mIsAbsolutelyPositioningEnabled = aIsEnabled;
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLEditor::RelativeChangeElementZIndex(nsIDOMElement * aElement,
|
|
int32_t aChange,
|
|
int32_t * aReturn)
|
|
{
|
|
NS_ENSURE_ARG_POINTER(aElement);
|
|
NS_ENSURE_ARG_POINTER(aReturn);
|
|
if (!aChange) // early way out, no change
|
|
return NS_OK;
|
|
|
|
int32_t zIndex;
|
|
nsresult res = GetElementZIndex(aElement, &zIndex);
|
|
NS_ENSURE_SUCCESS(res, res);
|
|
|
|
zIndex = std::max(zIndex + aChange, 0);
|
|
SetElementZIndex(aElement, zIndex);
|
|
*aReturn = zIndex;
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLEditor::SetElementZIndex(nsIDOMElement* aElement, int32_t aZindex)
|
|
{
|
|
nsCOMPtr<Element> element = do_QueryInterface(aElement);
|
|
NS_ENSURE_ARG_POINTER(element);
|
|
|
|
nsAutoString zIndexStr;
|
|
zIndexStr.AppendInt(aZindex);
|
|
|
|
mHTMLCSSUtils->SetCSSProperty(*element, *nsGkAtoms::z_index, zIndexStr);
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLEditor::RelativeChangeZIndex(int32_t aChange)
|
|
{
|
|
nsAutoEditBatch beginBatching(this);
|
|
nsAutoRules beginRulesSniffing(this,
|
|
(aChange < 0) ? EditAction::decreaseZIndex :
|
|
EditAction::increaseZIndex,
|
|
nsIEditor::eNext);
|
|
|
|
// brade: can we get rid of this comment?
|
|
// Find out if the selection is collapsed:
|
|
RefPtr<Selection> selection = GetSelection();
|
|
NS_ENSURE_TRUE(selection, NS_ERROR_NULL_POINTER);
|
|
nsTextRulesInfo ruleInfo(aChange < 0 ? EditAction::decreaseZIndex :
|
|
EditAction::increaseZIndex);
|
|
bool cancel, handled;
|
|
// Protect the edit rules object from dying
|
|
nsCOMPtr<nsIEditRules> kungFuDeathGrip(mRules);
|
|
nsresult res = mRules->WillDoAction(selection, &ruleInfo, &cancel, &handled);
|
|
if (cancel || NS_FAILED(res))
|
|
return res;
|
|
|
|
return mRules->DidDoAction(selection, &ruleInfo, res);
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLEditor::GetElementZIndex(nsIDOMElement * aElement,
|
|
int32_t * aZindex)
|
|
{
|
|
nsCOMPtr<Element> element = do_QueryInterface(aElement);
|
|
NS_ENSURE_STATE(element || !aElement);
|
|
nsAutoString zIndexStr;
|
|
*aZindex = 0;
|
|
|
|
nsresult res = mHTMLCSSUtils->GetSpecifiedProperty(*element,
|
|
*nsGkAtoms::z_index,
|
|
zIndexStr);
|
|
NS_ENSURE_SUCCESS(res, res);
|
|
if (zIndexStr.EqualsLiteral("auto")) {
|
|
// we have to look at the positioned ancestors
|
|
// cf. CSS 2 spec section 9.9.1
|
|
nsCOMPtr<nsIDOMNode> parentNode;
|
|
res = aElement->GetParentNode(getter_AddRefs(parentNode));
|
|
NS_ENSURE_SUCCESS(res, res);
|
|
nsCOMPtr<nsINode> node = do_QueryInterface(parentNode);
|
|
nsAutoString positionStr;
|
|
while (node && zIndexStr.EqualsLiteral("auto") &&
|
|
!node->IsHTMLElement(nsGkAtoms::body)) {
|
|
res = mHTMLCSSUtils->GetComputedProperty(*node, *nsGkAtoms::position,
|
|
positionStr);
|
|
NS_ENSURE_SUCCESS(res, res);
|
|
if (positionStr.EqualsLiteral("absolute")) {
|
|
// ah, we found one, what's its z-index ? If its z-index is auto,
|
|
// we have to continue climbing the document's tree
|
|
res = mHTMLCSSUtils->GetComputedProperty(*node, *nsGkAtoms::z_index,
|
|
zIndexStr);
|
|
NS_ENSURE_SUCCESS(res, res);
|
|
}
|
|
node = node->GetParentNode();
|
|
}
|
|
}
|
|
|
|
if (!zIndexStr.EqualsLiteral("auto")) {
|
|
nsresult errorCode;
|
|
*aZindex = zIndexStr.ToInteger(&errorCode);
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
already_AddRefed<Element>
|
|
nsHTMLEditor::CreateGrabber(nsINode* aParentNode)
|
|
{
|
|
// let's create a grabber through the element factory
|
|
nsCOMPtr<nsIDOMElement> retDOM;
|
|
CreateAnonymousElement(NS_LITERAL_STRING("span"), GetAsDOMNode(aParentNode),
|
|
NS_LITERAL_STRING("mozGrabber"), false,
|
|
getter_AddRefs(retDOM));
|
|
|
|
NS_ENSURE_TRUE(retDOM, nullptr);
|
|
|
|
// add the mouse listener so we can detect a click on a resizer
|
|
nsCOMPtr<nsIDOMEventTarget> evtTarget(do_QueryInterface(retDOM));
|
|
evtTarget->AddEventListener(NS_LITERAL_STRING("mousedown"),
|
|
mEventListener, false);
|
|
|
|
nsCOMPtr<Element> ret = do_QueryInterface(retDOM);
|
|
return ret.forget();
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLEditor::RefreshGrabber()
|
|
{
|
|
NS_ENSURE_TRUE(mAbsolutelyPositionedObject, NS_ERROR_NULL_POINTER);
|
|
|
|
nsresult res = GetPositionAndDimensions(static_cast<nsIDOMElement*>(GetAsDOMNode(mAbsolutelyPositionedObject)),
|
|
mPositionedObjectX,
|
|
mPositionedObjectY,
|
|
mPositionedObjectWidth,
|
|
mPositionedObjectHeight,
|
|
mPositionedObjectBorderLeft,
|
|
mPositionedObjectBorderTop,
|
|
mPositionedObjectMarginLeft,
|
|
mPositionedObjectMarginTop);
|
|
|
|
NS_ENSURE_SUCCESS(res, res);
|
|
|
|
SetAnonymousElementPosition(mPositionedObjectX+12,
|
|
mPositionedObjectY-14,
|
|
static_cast<nsIDOMElement*>(GetAsDOMNode(mGrabber)));
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLEditor::HideGrabber()
|
|
{
|
|
nsresult res = mAbsolutelyPositionedObject->UnsetAttr(kNameSpaceID_None,
|
|
nsGkAtoms::_moz_abspos,
|
|
true);
|
|
NS_ENSURE_SUCCESS(res, res);
|
|
|
|
mAbsolutelyPositionedObject = nullptr;
|
|
NS_ENSURE_TRUE(mGrabber, NS_ERROR_NULL_POINTER);
|
|
|
|
// get the presshell's document observer interface.
|
|
nsCOMPtr<nsIPresShell> ps = GetPresShell();
|
|
// We allow the pres shell to be null; when it is, we presume there
|
|
// are no document observers to notify, but we still want to
|
|
// UnbindFromTree.
|
|
|
|
nsCOMPtr<nsIContent> parentContent = mGrabber->GetParent();
|
|
NS_ENSURE_TRUE(parentContent, NS_ERROR_NULL_POINTER);
|
|
|
|
DeleteRefToAnonymousNode(static_cast<nsIDOMElement*>(GetAsDOMNode(mGrabber)), parentContent, ps);
|
|
mGrabber = nullptr;
|
|
DeleteRefToAnonymousNode(static_cast<nsIDOMElement*>(GetAsDOMNode(mPositioningShadow)), parentContent, ps);
|
|
mPositioningShadow = nullptr;
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLEditor::ShowGrabberOnElement(nsIDOMElement * aElement)
|
|
{
|
|
nsCOMPtr<Element> element = do_QueryInterface(aElement);
|
|
NS_ENSURE_ARG_POINTER(element);
|
|
|
|
if (NS_WARN_IF(!IsDescendantOfEditorRoot(element))) {
|
|
return NS_ERROR_UNEXPECTED;
|
|
}
|
|
|
|
if (mGrabber) {
|
|
NS_ERROR("call HideGrabber first");
|
|
return NS_ERROR_UNEXPECTED;
|
|
}
|
|
|
|
nsAutoString classValue;
|
|
nsresult res = CheckPositionedElementBGandFG(aElement, classValue);
|
|
NS_ENSURE_SUCCESS(res, res);
|
|
|
|
res = element->SetAttr(kNameSpaceID_None, nsGkAtoms::_moz_abspos,
|
|
classValue, true);
|
|
NS_ENSURE_SUCCESS(res, res);
|
|
|
|
// first, let's keep track of that element...
|
|
mAbsolutelyPositionedObject = element;
|
|
|
|
mGrabber = CreateGrabber(element->GetParentNode());
|
|
NS_ENSURE_TRUE(mGrabber, NS_ERROR_FAILURE);
|
|
|
|
// and set its position
|
|
return RefreshGrabber();
|
|
}
|
|
|
|
nsresult
|
|
nsHTMLEditor::StartMoving(nsIDOMElement *aHandle)
|
|
{
|
|
nsCOMPtr<nsINode> parentNode = mGrabber->GetParentNode();
|
|
|
|
// now, let's create the resizing shadow
|
|
mPositioningShadow = CreateShadow(GetAsDOMNode(parentNode),
|
|
static_cast<nsIDOMElement*>(GetAsDOMNode(mAbsolutelyPositionedObject)));
|
|
NS_ENSURE_TRUE(mPositioningShadow, NS_ERROR_FAILURE);
|
|
nsresult res = SetShadowPosition(mPositioningShadow,
|
|
mAbsolutelyPositionedObject,
|
|
mPositionedObjectX, mPositionedObjectY);
|
|
NS_ENSURE_SUCCESS(res,res);
|
|
|
|
// make the shadow appear
|
|
mPositioningShadow->UnsetAttr(kNameSpaceID_None, nsGkAtoms::_class, true);
|
|
|
|
// position it
|
|
mHTMLCSSUtils->SetCSSPropertyPixels(*mPositioningShadow, *nsGkAtoms::width,
|
|
mPositionedObjectWidth);
|
|
mHTMLCSSUtils->SetCSSPropertyPixels(*mPositioningShadow, *nsGkAtoms::height,
|
|
mPositionedObjectHeight);
|
|
|
|
mIsMoving = true;
|
|
return res;
|
|
}
|
|
|
|
void
|
|
nsHTMLEditor::SnapToGrid(int32_t & newX, int32_t & newY)
|
|
{
|
|
if (mSnapToGridEnabled && mGridSize) {
|
|
newX = (int32_t) floor( ((float)newX / (float)mGridSize) + 0.5f ) * mGridSize;
|
|
newY = (int32_t) floor( ((float)newY / (float)mGridSize) + 0.5f ) * mGridSize;
|
|
}
|
|
}
|
|
|
|
nsresult
|
|
nsHTMLEditor::GrabberClicked()
|
|
{
|
|
// add a mouse move listener to the editor
|
|
nsresult res = NS_OK;
|
|
if (!mMouseMotionListenerP) {
|
|
mMouseMotionListenerP = new ResizerMouseMotionListener(this);
|
|
if (!mMouseMotionListenerP) {return NS_ERROR_NULL_POINTER;}
|
|
|
|
nsCOMPtr<nsIDOMEventTarget> piTarget = GetDOMEventTarget();
|
|
NS_ENSURE_TRUE(piTarget, NS_ERROR_FAILURE);
|
|
|
|
res = piTarget->AddEventListener(NS_LITERAL_STRING("mousemove"),
|
|
mMouseMotionListenerP,
|
|
false, false);
|
|
NS_ASSERTION(NS_SUCCEEDED(res),
|
|
"failed to register mouse motion listener");
|
|
}
|
|
mGrabberClicked = true;
|
|
return res;
|
|
}
|
|
|
|
nsresult
|
|
nsHTMLEditor::EndMoving()
|
|
{
|
|
if (mPositioningShadow) {
|
|
nsCOMPtr<nsIPresShell> ps = GetPresShell();
|
|
NS_ENSURE_TRUE(ps, NS_ERROR_NOT_INITIALIZED);
|
|
|
|
nsCOMPtr<nsIContent> parentContent = mGrabber->GetParent();
|
|
NS_ENSURE_TRUE(parentContent, NS_ERROR_FAILURE);
|
|
|
|
DeleteRefToAnonymousNode(static_cast<nsIDOMElement*>(GetAsDOMNode(mPositioningShadow)),
|
|
parentContent, ps);
|
|
|
|
mPositioningShadow = nullptr;
|
|
}
|
|
nsCOMPtr<nsIDOMEventTarget> piTarget = GetDOMEventTarget();
|
|
|
|
if (piTarget && mMouseMotionListenerP) {
|
|
#ifdef DEBUG
|
|
nsresult res =
|
|
#endif
|
|
piTarget->RemoveEventListener(NS_LITERAL_STRING("mousemove"),
|
|
mMouseMotionListenerP,
|
|
false);
|
|
NS_ASSERTION(NS_SUCCEEDED(res), "failed to remove mouse motion listener");
|
|
}
|
|
mMouseMotionListenerP = nullptr;
|
|
|
|
mGrabberClicked = false;
|
|
mIsMoving = false;
|
|
RefPtr<Selection> selection = GetSelection();
|
|
if (!selection) {
|
|
return NS_ERROR_NOT_INITIALIZED;
|
|
}
|
|
return CheckSelectionStateForAnonymousButtons(selection);
|
|
}
|
|
nsresult
|
|
nsHTMLEditor::SetFinalPosition(int32_t aX, int32_t aY)
|
|
{
|
|
nsresult res = EndMoving();
|
|
NS_ENSURE_SUCCESS(res, res);
|
|
|
|
// we have now to set the new width and height of the resized object
|
|
// we don't set the x and y position because we don't control that in
|
|
// a normal HTML layout
|
|
int32_t newX = mPositionedObjectX + aX - mOriginalX - (mPositionedObjectBorderLeft+mPositionedObjectMarginLeft);
|
|
int32_t newY = mPositionedObjectY + aY - mOriginalY - (mPositionedObjectBorderTop+mPositionedObjectMarginTop);
|
|
|
|
SnapToGrid(newX, newY);
|
|
|
|
nsAutoString x, y;
|
|
x.AppendInt(newX);
|
|
y.AppendInt(newY);
|
|
|
|
// we want one transaction only from a user's point of view
|
|
nsAutoEditBatch batchIt(this);
|
|
|
|
nsCOMPtr<Element> absolutelyPositionedObject =
|
|
do_QueryInterface(mAbsolutelyPositionedObject);
|
|
NS_ENSURE_STATE(absolutelyPositionedObject);
|
|
mHTMLCSSUtils->SetCSSPropertyPixels(*absolutelyPositionedObject,
|
|
*nsGkAtoms::top, newY);
|
|
mHTMLCSSUtils->SetCSSPropertyPixels(*absolutelyPositionedObject,
|
|
*nsGkAtoms::left, newX);
|
|
// keep track of that size
|
|
mPositionedObjectX = newX;
|
|
mPositionedObjectY = newY;
|
|
|
|
return RefreshResizers();
|
|
}
|
|
|
|
void
|
|
nsHTMLEditor::AddPositioningOffset(int32_t & aX, int32_t & aY)
|
|
{
|
|
// Get the positioning offset
|
|
int32_t positioningOffset =
|
|
Preferences::GetInt("editor.positioning.offset", 0);
|
|
|
|
aX += positioningOffset;
|
|
aY += positioningOffset;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLEditor::AbsolutelyPositionElement(nsIDOMElement* aElement,
|
|
bool aEnabled)
|
|
{
|
|
nsCOMPtr<Element> element = do_QueryInterface(aElement);
|
|
NS_ENSURE_ARG_POINTER(element);
|
|
|
|
nsAutoString positionStr;
|
|
mHTMLCSSUtils->GetComputedProperty(*element, *nsGkAtoms::position,
|
|
positionStr);
|
|
bool isPositioned = (positionStr.EqualsLiteral("absolute"));
|
|
|
|
// nothing to do if the element is already in the state we want
|
|
if (isPositioned == aEnabled)
|
|
return NS_OK;
|
|
|
|
nsAutoEditBatch batchIt(this);
|
|
|
|
if (aEnabled) {
|
|
int32_t x, y;
|
|
GetElementOrigin(aElement, x, y);
|
|
|
|
mHTMLCSSUtils->SetCSSProperty(*element, *nsGkAtoms::position,
|
|
NS_LITERAL_STRING("absolute"));
|
|
|
|
AddPositioningOffset(x, y);
|
|
SnapToGrid(x, y);
|
|
SetElementPosition(aElement, x, y);
|
|
|
|
// we may need to create a br if the positioned element is alone in its
|
|
// container
|
|
nsCOMPtr<nsINode> element = do_QueryInterface(aElement);
|
|
NS_ENSURE_STATE(element);
|
|
|
|
nsINode* parentNode = element->GetParentNode();
|
|
if (parentNode->GetChildCount() == 1) {
|
|
nsCOMPtr<nsIDOMNode> brNode;
|
|
nsresult res = CreateBR(parentNode->AsDOMNode(), 0, address_of(brNode));
|
|
NS_ENSURE_SUCCESS(res, res);
|
|
}
|
|
}
|
|
else {
|
|
mHTMLCSSUtils->RemoveCSSProperty(*element, *nsGkAtoms::position,
|
|
EmptyString());
|
|
mHTMLCSSUtils->RemoveCSSProperty(*element, *nsGkAtoms::top,
|
|
EmptyString());
|
|
mHTMLCSSUtils->RemoveCSSProperty(*element, *nsGkAtoms::left,
|
|
EmptyString());
|
|
mHTMLCSSUtils->RemoveCSSProperty(*element, *nsGkAtoms::z_index,
|
|
EmptyString());
|
|
|
|
if (!nsHTMLEditUtils::IsImage(aElement)) {
|
|
mHTMLCSSUtils->RemoveCSSProperty(*element, *nsGkAtoms::width,
|
|
EmptyString());
|
|
mHTMLCSSUtils->RemoveCSSProperty(*element, *nsGkAtoms::height,
|
|
EmptyString());
|
|
}
|
|
|
|
nsCOMPtr<dom::Element> element = do_QueryInterface(aElement);
|
|
if (element && element->IsHTMLElement(nsGkAtoms::div) &&
|
|
!HasStyleOrIdOrClass(element)) {
|
|
RefPtr<nsHTMLEditRules> htmlRules = static_cast<nsHTMLEditRules*>(mRules.get());
|
|
NS_ENSURE_TRUE(htmlRules, NS_ERROR_FAILURE);
|
|
nsresult res = htmlRules->MakeSureElemStartsOrEndsOnCR(aElement);
|
|
NS_ENSURE_SUCCESS(res, res);
|
|
res = RemoveContainer(element);
|
|
NS_ENSURE_SUCCESS(res, res);
|
|
}
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLEditor::SetSnapToGridEnabled(bool aEnabled)
|
|
{
|
|
mSnapToGridEnabled = aEnabled;
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLEditor::GetSnapToGridEnabled(bool * aIsEnabled)
|
|
{
|
|
*aIsEnabled = mSnapToGridEnabled;
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLEditor::SetGridSize(uint32_t aSize)
|
|
{
|
|
mGridSize = aSize;
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLEditor::GetGridSize(uint32_t * aSize)
|
|
{
|
|
*aSize = mGridSize;
|
|
return NS_OK;
|
|
}
|
|
|
|
// self-explanatory
|
|
NS_IMETHODIMP
|
|
nsHTMLEditor::SetElementPosition(nsIDOMElement *aElement, int32_t aX, int32_t aY)
|
|
{
|
|
nsCOMPtr<Element> element = do_QueryInterface(aElement);
|
|
NS_ENSURE_STATE(element);
|
|
nsAutoEditBatch batchIt(this);
|
|
|
|
mHTMLCSSUtils->SetCSSPropertyPixels(*element, *nsGkAtoms::left, aX);
|
|
mHTMLCSSUtils->SetCSSPropertyPixels(*element, *nsGkAtoms::top, aY);
|
|
return NS_OK;
|
|
}
|
|
|
|
// self-explanatory
|
|
NS_IMETHODIMP
|
|
nsHTMLEditor::GetPositionedElement(nsIDOMElement ** aReturn)
|
|
{
|
|
nsCOMPtr<nsIDOMElement> ret =
|
|
static_cast<nsIDOMElement*>(GetAsDOMNode(mAbsolutelyPositionedObject));
|
|
ret.forget(aReturn);
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult
|
|
nsHTMLEditor::CheckPositionedElementBGandFG(nsIDOMElement * aElement,
|
|
nsAString & aReturn)
|
|
{
|
|
// we are going to outline the positioned element and bring it to the
|
|
// front to overlap any other element intersecting with it. But
|
|
// first, let's see what's the background and foreground colors of the
|
|
// positioned element.
|
|
// if background-image computed value is 'none,
|
|
// If the background color is 'auto' and R G B values of the foreground are
|
|
// each above #d0, use a black background
|
|
// If the background color is 'auto' and at least one of R G B values of
|
|
// the foreground is below #d0, use a white background
|
|
// Otherwise don't change background/foreground
|
|
nsCOMPtr<Element> element = do_QueryInterface(aElement);
|
|
NS_ENSURE_STATE(element || !aElement);
|
|
|
|
aReturn.Truncate();
|
|
|
|
nsAutoString bgImageStr;
|
|
nsresult res =
|
|
mHTMLCSSUtils->GetComputedProperty(*element, *nsGkAtoms::background_image,
|
|
bgImageStr);
|
|
NS_ENSURE_SUCCESS(res, res);
|
|
if (bgImageStr.EqualsLiteral("none")) {
|
|
nsAutoString bgColorStr;
|
|
res =
|
|
mHTMLCSSUtils->GetComputedProperty(*element, *nsGkAtoms::backgroundColor,
|
|
bgColorStr);
|
|
NS_ENSURE_SUCCESS(res, res);
|
|
if (bgColorStr.EqualsLiteral("transparent")) {
|
|
RefPtr<nsComputedDOMStyle> cssDecl =
|
|
mHTMLCSSUtils->GetComputedStyle(element);
|
|
NS_ENSURE_STATE(cssDecl);
|
|
|
|
// from these declarations, get the one we want and that one only
|
|
ErrorResult error;
|
|
RefPtr<dom::CSSValue> cssVal = cssDecl->GetPropertyCSSValue(NS_LITERAL_STRING("color"), error);
|
|
NS_ENSURE_TRUE(!error.Failed(), error.StealNSResult());
|
|
|
|
nsROCSSPrimitiveValue* val = cssVal->AsPrimitiveValue();
|
|
NS_ENSURE_TRUE(val, NS_ERROR_FAILURE);
|
|
|
|
if (nsIDOMCSSPrimitiveValue::CSS_RGBCOLOR == val->PrimitiveType()) {
|
|
nsDOMCSSRGBColor* rgbVal = val->GetRGBColorValue(error);
|
|
NS_ENSURE_TRUE(!error.Failed(), error.StealNSResult());
|
|
float r = rgbVal->Red()->
|
|
GetFloatValue(nsIDOMCSSPrimitiveValue::CSS_NUMBER, error);
|
|
NS_ENSURE_TRUE(!error.Failed(), error.StealNSResult());
|
|
float g = rgbVal->Green()->
|
|
GetFloatValue(nsIDOMCSSPrimitiveValue::CSS_NUMBER, error);
|
|
NS_ENSURE_TRUE(!error.Failed(), error.StealNSResult());
|
|
float b = rgbVal->Blue()->
|
|
GetFloatValue(nsIDOMCSSPrimitiveValue::CSS_NUMBER, error);
|
|
NS_ENSURE_TRUE(!error.Failed(), error.StealNSResult());
|
|
if (r >= BLACK_BG_RGB_TRIGGER &&
|
|
g >= BLACK_BG_RGB_TRIGGER &&
|
|
b >= BLACK_BG_RGB_TRIGGER)
|
|
aReturn.AssignLiteral("black");
|
|
else
|
|
aReturn.AssignLiteral("white");
|
|
return NS_OK;
|
|
}
|
|
}
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|